机器学习首先面临的一个问题就是准备数据,数据的来源大概有这么几种:公司积累数据,购买,交换,政府机构及企业公开的数据,通过爬虫从网上抓取。本篇介绍怎么写一个爬虫从网上抓取公开的数据。
很多语言都可以写爬虫,但是不同语言的难易程度不同,Python作为一种解释型的胶水语言,上手简单、入门容易,标准库齐全,还有丰富的各种开源库,语言本身提供了很多提高开发效率的语法糖,开发效率高,总之“人生苦短,快用Python”(Life is short, you need Python!)。在Web网站开发,科学计算,数据挖掘/分析,人工智能等很多领域广泛使用。
开发环境配置,Python3.5.2,Scrapy1.2.1,使用pip安装scrapy,命令:pip3 install Scrapy,此命令在Mac下会自动安装Scrapy的依赖包,安装过程中如果出现网络超时,多试几次。
创建工程
首先创建一个Scrapy工程,工程名为:kiwi,命令:scrapy startproject kiwi,将创建一些文件夹和文件模板。
定义数据结构
settings.py是一些设置信息,items.py用来保存解析出来的数据,在此文件里定义一些数据结构,示例代码:
import scrapy
class AuthorInfo(scrapy.Item):
    authorName = scrapy.Field()  # 作者昵称
    authorUrl = scrapy.Field()  # 作者Url
class ReplyItem(scrapy.Item):
    content = scrapy.Field()  # 回复内容
    time = scrapy.Field()  # 发布时间
    author = scrapy.Field() # 回复人(AuthorInfo)
class TopicItem(scrapy.Item):
    title = scrapy.Field() # 帖子标题
    url = scrapy.Field() # 帖子页面Url
    content = scrapy.Field() # 帖子内容
    time = scrapy.Field()  # 发布时间
    author = scrapy.Field() # 发帖人(AuthorInfo)
    reply = scrapy.Field() # 回复列表(ReplyItem list)
    replyCount = scrapy.Field() # 回复条数
上面TopicItem中嵌套了AuthorInfo和ReplyItem list,但是初始化类型必须是scrapy.Field(),注意这三个类都需要从scrapy.Item继续。
创建爬虫蜘蛛
工程目录spiders下的kiwi_spider.py文件是爬虫蜘蛛代码,爬虫代码写在这个文件里。示例以爬豆瓣群组里的帖子和回复为例。
# -*- coding: utf-8 -*-
from scrapy.selector import Selector
from scrapy.spiders import CrawlSpider, Rule
from scrapy.linkextractors import LinkExtractor
from kiwi.items import TopicItem, AuthorInfo, ReplyItem
class KiwiSpider(CrawlSpider):
    name = "kiwi"
    allowed_domains = ["douban.com"]
    anchorTitleXPath = 'a/text()'
    anchorHrefXPath = 'a/@href'
    start_urls = [
        "https://www.douban.com/group/topic/90895393/?start=0",
    ]
    rules = (
        Rule(
            LinkExtractor(allow=(r'/group/[^/]+/discussion\?start=\d+',)),
                callback='parse_topic_list',
                follow=True
        ),
        Rule(
            LinkExtractor(allow=(r'/group/topic/\d+/$',)),  # 帖子内容页面
                callback='parse_topic_content',
                follow=True
        ),
        Rule(
            LinkExtractor(allow=(r'/group/topic/\d+/\?start=\d+',)), # 帖子内容页面
                callback='parse_topic_content',
                follow=True
        ),
    )
    # 帖子详情页面
    def parse_topic_content(self, response):
        # 标题XPath
        titleXPath = '//html/head/title/text()'
        # 帖子内容XPath
        contentXPath = '//div[@class="topic-content"]/p/text()'
        # 发帖时间XPath
        timeXPath = '//div[@class="topic-doc"]/h3/span[@class="color-green"]/text()'
        # 发帖人XPath
        authorXPath = '//div[@class="topic-doc"]/h3/span[@class="from"]'
        item = TopicItem()
        # 当前页面Url
        item['url'] = response.url
        # 标题
        titleFragment = Selector(response).xpath(titleXPath)
        item['title'] = str(titleFragment.extract()[0]).strip()
        # 帖子内容
        contentFragment = Selector(response).xpath(contentXPath)
        strs = [line.extract().strip() for line in contentFragment]
        item['content'] = '\n'.join(strs)
        # 发帖时间
        timeFragment = Selector(response).xpath(timeXPath)
        if timeFragment:
            item['time'] = timeFragment[0].extract()
        # 发帖人信息
        authorInfo = AuthorInfo()
        authorFragment = Selector(response).xpath(authorXPath)
        if authorFragment:
            authorInfo['authorName'] = authorFragment[0].xpath(self.anchorTitleXPath).extract()[0]
            authorInfo['authorUrl'] = authorFragment[0].xpath(self.anchorHrefXPath).extract()[0]
        item['author'] = dict(authorInfo)
        # 回复列表XPath
        replyRootXPath = r'//div[@class="reply-doc content"]'
        # 回复时间XPath
        replyTimeXPath = r'div[@class="bg-img-green"]/h4/span[@class="pubtime"]/text()'
        # 回复人XPath
        replyAuthorXPath = r'div[@class="bg-img-green"]/h4'
        replies = []
        itemsFragment = Selector(response).xpath(replyRootXPath)
        for replyItemXPath in itemsFragment:
            replyItem = ReplyItem()
            # 回复内容
            contents = replyItemXPath.xpath('p/text()')
            strs = [line.extract().strip() for line in contents]
            replyItem['content'] = '\n'.join(strs)
            # 回复时间
            timeFragment = replyItemXPath.xpath(replyTimeXPath)
            if timeFragment:
                replyItem['time'] = timeFragment[0].extract()
            # 回复人
            replyAuthorInfo = AuthorInfo()
            authorFragment = replyItemXPath.xpath(replyAuthorXPath)
            if authorFragment:
                replyAuthorInfo['authorName'] = authorFragment[0].xpath(self.anchorTitleXPath).extract()[0]
                replyAuthorInfo['authorUrl'] = authorFragment[0].xpath(self.anchorHrefXPath).extract()[0]
            replyItem['author'] = dict(replyAuthorInfo)
            # 添加进回复列表
            replies.append(dict(replyItem))
        item['reply'] = replies
        yield item
    # 帖子列表页面
    def parse_topic_list(self, response):
        # 帖子列表XPath(跳过表头行)
        topicRootXPath = r'//table[@class="olt"]/tr[position()>1]'
        # 单条帖子条目XPath
        titleXPath = r'td[@class="title"]'
        # 发帖人XPath
        authorXPath = r'td[2]'
        # 回复条数XPath
        replyCountXPath = r'td[3]/text()'
        # 发帖时间XPath
        timeXPath = r'td[@class="time"]/text()'
        topicsPath = Selector(response).xpath(topicRootXPath)
        for topicItemPath in topicsPath:
            item = TopicItem()
            titlePath = topicItemPath.xpath(titleXPath)
            item['title'] = titlePath.xpath(self.anchorTitleXPath).extract()[0]
            item['url'] = titlePath.xpath(self.anchorHrefXPath).extract()[0]
            # 发帖时间
            timePath = topicItemPath.xpath(timeXPath)
            if timePath:
                item['time'] = timePath[0].extract()
                # 发帖人
                authorPath = topicItemPath.xpath(authorXPath)
                authInfo = AuthorInfo()
                authInfo['authorName'] = authorPath[0].xpath(self.anchorTitleXPath).extract()[0]
                authInfo['authorUrl'] = authorPath[0].xpath(self.anchorHrefXPath).extract()[0]
                item['author'] = dict(authInfo)
                # 回复条数
                replyCountPath = topicItemPath.xpath(replyCountXPath)
                item['replyCount'] = replyCountPath[0].extract()
            item['content'] = ''
            yield item
    parse_start_url = parse_topic_content
特别注意
1、KiwiSpider需要改成从CrawlSpider类继承,模板生成的代码是从Spider继承的,那样的话不会去爬rules里的页面。
2、parse_start_url = parse_topic_list 是定义入口函数,从CrawlSpider类的代码里可以看到parse函数回调的是parse_start_url函数,子类可以重写这个函数,也可以像上面代码那样给它赋值一个新函数。
3、start_urls里是入口网址,可以添加多个网址。
4、rules里定义在抓取到的网页中哪些网址需要进去爬,规则和对应的回调函数,规则用正则表达式写。上面的示例代码,定义了继续抓取帖子详情首页及分页。
5、注意代码里用dict()包装的部分,items.py文件里定义数据结构的时候,author属性实际需要的是AuthorInfo类型,赋值的时候必须用dict包装起来,item[‘author’] = authInfo 赋值会报Error。
6、提取内容的时候利用XPath取出需要的内容,有开发过程中可以利用浏览器提供的工具查看XPath,比如Firefox 浏览器中的FireBug、FirePath插件,对于https://www.douban.com/group/python/discussion?start=0这个页面,XPath规则“//td[@class=“title”]”可以获取到帖子标题列表,示例:
上图红框中可以输入XPath规则,方便测试XPath的规则是否符合要求。新版Firefox可以安装 Try XPath 这个插件 查看XPath,Chrome浏览器可以安装 XPath Helper 插件。
使用随机UserAgent
为了让网站看来更像是正常的浏览器访问,可以写一个Middleware提供随机的User-Agent,在工程根目录下添加文件useragentmiddleware.py,示例代码:
# -*-coding:utf-8-*-
import random
from scrapy.downloadermiddlewares.useragent import UserAgentMiddleware
class RotateUserAgentMiddleware(UserAgentMiddleware):
    def __init__(self, user_agent=''):
        self.user_agent = user_agent
    def process_request(self, request, spider):
        ua = random.choice(self.user_agent_list)
        if ua:
            request.headers.setdefault('User-Agent', ua)
    # for more user agent strings,you can find it in http://www.useragentstring.com/pages/useragentstring.php
    user_agent_list = [ \
        "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.1 (KHTML, like Gecko) Chrome/22.0.1207.1 Safari/537.1" \
        "Mozilla/5.0 (X11; CrOS i686 2268.111.0) AppleWebKit/536.11 (KHTML, like Gecko) Chrome/20.0.1132.57 Safari/536.11", \
        "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.6 (KHTML, like Gecko) Chrome/20.0.1092.0 Safari/536.6", \
        "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.6 (KHTML, like Gecko) Chrome/20.0.1090.0 Safari/536.6", \
        "Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/537.1 (KHTML, like Gecko) Chrome/19.77.34.5 Safari/537.1", \
        "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/536.5 (KHTML, like Gecko) Chrome/19.0.1084.9 Safari/536.5", \
        "Mozilla/5.0 (Windows NT 6.0) AppleWebKit/536.5 (KHTML, like Gecko) Chrome/19.0.1084.36 Safari/536.5", \
        "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1063.0 Safari/536.3", \
        "Mozilla/5.0 (Windows NT 5.1) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1063.0 Safari/536.3", \
        "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_8_0) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1063.0 Safari/536.3", \
        "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1062.0 Safari/536.3", \
        "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1062.0 Safari/536.3", \
        "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.1 Safari/536.3", \
        "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.1 Safari/536.3", \
        "Mozilla/5.0 (Windows NT 6.1) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.1 Safari/536.3", \
        "Mozilla/5.0 (Windows NT 6.2) AppleWebKit/536.3 (KHTML, like Gecko) Chrome/19.0.1061.0 Safari/536.3", \
        "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/535.24 (KHTML, like Gecko) Chrome/19.0.1055.1 Safari/535.24", \
        "Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/535.24 (KHTML, like Gecko) Chrome/19.0.1055.1 Safari/535.24"
    ]
修改settings.py,添加下面的设置,
DOWNLOADER_MIDDLEWARES = {
   'kiwi.useragentmiddleware.RotateUserAgentMiddleware': 1,
}
同时禁用cookie,COOKIES_ENABLED = False。
运行爬虫
切换到工程根目录,输入命令:scrapy crawl kiwi,console窗口可以看到打印出来的数据,或者使用命令“scrapy crawl kiwi -o result.json -t json”将结果保存到文件里。
怎么抓取用JS代码动态输出的网页数据
上面的例子对由执行js代码输出数据的页面不适用,好在Python的工具库多,可以安装phantomjs这个工具,从官网下载解压即可。下面以抓取 http://www.kjj.com/index_kfjj.html 这个网页的基金净值数据为例,这个页面的数据是由js代码动态输出的,js代码执行之后才会输出基金净值列表。fund_spider.py代码
# -*- coding: utf-8 -*-
from scrapy.selector import Selector
from datetime import  datetime
from selenium import webdriver
from fundequity import FundEquity
'''
遇到不懂的问题?Python学习交流群:821460695满足你的需求,资料都已经上传群文件,可以自行下载!
'''
class PageSpider(object):
    def __init__(self):
        phantomjsPath = "/Library/Frameworks/Python.framework/Versions/3.5/phantomjs/bin/phantomjs"
        cap = webdriver.DesiredCapabilities.PHANTOMJS
        cap["phantomjs.page.settings.resourceTimeout"] = 1000
        cap["phantomjs.page.settings.loadImages"] = False
        cap["phantomjs.page.settings.disk-cache"] = False
        self.driver = webdriver.PhantomJS(executable_path=phantomjsPath, desired_capabilities=cap)
    def fetchPage(self, url):
        self.driver.get(url)
        html = self.driver.page_source
        return html
    def parse(self, html):
        fundListXPath = r'//div[@id="maininfo_all"]/table[@id="ilist"]/tbody/tr[position()>1]'
        itemsFragment = Selector(text=html).xpath(fundListXPath)
        for itemXPath in itemsFragment:
            attrXPath = itemXPath.xpath(r'td[1]/text()')
            text = attrXPath[0].extract().strip()
            if text != "-":
                fe = FundEquity()
                fe.serial = text
                attrXPath = itemXPath.xpath(r'td[2]/text()')
                text = attrXPath[0].extract().strip()
                fe.date = datetime.strptime(text, "%Y-%m-%d")
                attrXPath = itemXPath.xpath(r'td[3]/text()')
                text = attrXPath[0].extract().strip()
                fe.code = text
                attrXPath = itemXPath.xpath(r'td[4]/a/text()')
                text = attrXPath[0].extract().strip()
                fe.name = text
                attrXPath = itemXPath.xpath(r'td[5]/text()')
                text = attrXPath[0].extract().strip()
                fe.equity = text
                attrXPath = itemXPath.xpath(r'td[6]/text()')
                text = attrXPath[0].extract().strip()
                fe.accumulationEquity = text
                attrXPath = itemXPath.xpath(r'td[7]/font/text()')
                text = attrXPath[0].extract().strip()
                fe.increment = text
                attrXPath = itemXPath.xpath(r'td[8]/font/text()')
                text = attrXPath[0].extract().strip().strip('%')
                fe.growthRate = text
                attrXPath = itemXPath.xpath(r'td[9]/a/text()')
                if len(attrXPath) > 0:
                    text = attrXPath[0].extract().strip()
                    if text == "购买":
                        fe.canBuy = True
                    else:
                        fe.canBuy = False
                attrXPath = itemXPath.xpath(r'td[10]/font/text()')
                if len(attrXPath) > 0:
                    text = attrXPath[0].extract().strip()
                    if text == "赎回":
                        fe.canRedeem = True
                    else:
                        fe.canRedeem = False
                yield fe
    def __del__(self):
        self.driver.quit()
def test():
    spider = PageSpider()
    html = spider.fetchPage("http://www.kjj.com/index_kfjj.html")
    for item in spider.parse(html):
        print(item)
    del spider
if __name__ == "__main__":
    test()
# -*- coding: utf-8 -*-
from datetime import date
# 基金净值信息
class FundEquity(object):
    def __init__(self):
        # 类实例即对象的属性
        self.__serial = 0  # 序号
        self.__date = None  # 日期
        self.__code = ""  # 基金代码
        self.__name = ""  # 基金名称
        self.__equity = 0.0  # 单位净值
        self.__accumulationEquity = 0.0  # 累计净值
        self.__increment = 0.0  # 增长值
        self.__growthRate = 0.0  # 增长率
        self.__canBuy = False # 是否可以购买
        self.__canRedeem = True # 是否能赎回
    @property
    def serial(self):
        return self.__serial
    @serial.setter
    def serial(self, value):
        self.__serial = value
    @property
    def date(self):
        return self.__date
    @date.setter
    def date(self, value):
        # 数据检查
        if not isinstance(value, date):
            raise ValueError('date must be date type!')
        self.__date = value
    @property
    def code(self):
        return self.__code
    @code.setter
    def code(self, value):
        self.__code = value
    @property
    def name(self):
        return self.__name
    @name.setter
    def name(self, value):
        self.__name = value
    @property
    def equity(self):
        return self.__equity
    @equity.setter
    def equity(self, value):
        self.__equity = value
    @property
    def accumulationEquity(self):
        return self.__accumulationEquity
    @accumulationEquity.setter
    def accumulationEquity(self, value):
        self.__accumulationEquity = value
    @property
    def increment(self):
        return self.__increment
    @increment.setter
    def increment(self, value):
        self.__increment = value
    @property
    def growthRate(self):
        return self.__growthRate
    @growthRate.setter
    def growthRate(self, value):
        self.__growthRate = value
    @property
    def canBuy(self):
        return self.__canBuy
    @canBuy.setter
    def canBuy(self, value):
        self.__canBuy = value
    @property
    def canRedeem(self):
        return self.__canRedeem
    @canRedeem.setter
    def canRedeem(self, value):
        self.__canRedeem = value
    # 类似其它语言中的toString()函数
    def __str__(self):
        return '[serial:%s,date:%s,code:%s,name:%s,equity:%.4f,\
accumulationEquity:%.4f,increment:%.4f,growthRate:%.4f%%,canBuy:%s,canRedeem:%s]'\
               % (self.serial, self.date.strftime("%Y-%m-%d"), self.code, self.name, float(self.equity), \
                  float(self.accumulationEquity), float(self.increment), \
float(self.growthRate), self.canBuy, self.canRedeem)
上述代码中FundEquity类的属性值使用getter/setter函数方式定义的,这种方式可以对值进行检查。str(self)函数类似其它语言里的toString()。
在命令行运行fund_spider.py代码,console窗口会输出净值数据。