前言:本篇博客将爬取顶点小说网站全部小说、涉及到的问题有:Scrapy架构、断点续传问题、Mongodb数据库相关操作。

背景:

Python版本:Anaconda3

运行平台:Windows

IDE:PyCharm

数据库:MongoDB

浏览器工具: Chrome浏览器

前面的博客中已经对Scrapy作了相当多的介绍所以这里不再对Scrapy技术作过多的讲解。

一、爬虫准备工作:

此次我们爬取的是免费小说网站:顶点小说
http://www.23us.so/
我们要想把它全部的小说爬取下来,是不是得有全部
小说的链接?
我们看到顶点小说网站上有一个总排行榜。

Python爬虫爬起点章节 爬虫爬取小说_scrapy

点击进入后我们看到,这里有网站上所有的小说,一共有1144页,每页大约20本小说,算下来一共大约有两万两千多本,是一个庞大的数据量,并且小说的数量还在不断的增长中。
好!我们遇到了第一个问题,如何获取总排行榜中的页数呢?也就是现在的“1144”。
1、获取排行榜页面数:
最好的方法就是用Xpath。
我们先用F12审查元素,看到“1144”放在了“id”属性为“pagestats”的em节点中。

Python爬虫爬起点章节 爬虫爬取小说_mongodb_02

我们再用Scrapy Shell分析一下网页。
注意:Scrapy Shell是一个非常好的工具,我们在编写爬虫过程中,可以用它不断的测试我们编写的Xpath语句,非常方便。
输入命令:
scrapy shell "http://www.23us.so/top/allvisit_2.html"

然后就进入了scrapy shell

Python爬虫爬起点章节 爬虫爬取小说_网络爬虫_03

因为页数放在“id”属性为“pagestats”的em节点中,所以我们可以在shell中输入如下指令获取。
response.xpath('//*[@id="pagestats"]/text()').extract_first()

Python爬虫爬起点章节 爬虫爬取小说_mongodb_04

我们可以看到,Xpath一如既往的简单高效,页面数已经被截取下来了。

2、获取小说主页链接、小说名称:
接下来,我们遇到新的问题,如何获得每个页面上的小说的链接呢?我们再来看页面的HTML代码。

Python爬虫爬起点章节 爬虫爬取小说_scrapy_05

小说的链接放在了“a”节点里,而且这样的a节点区别其他的“a”节点的是,没有“title”属性。
所以我们用shell测试一下,输入命令:
response.xpath('//td/a[not(@title)]/@href').extract()

Python爬虫爬起点章节 爬虫爬取小说_python_06


我们看到,小说的链接地址我们抓到了。

同样还有小说名,
response.xpath('//td/a[not(@title)]/text()').extract()

Python爬虫爬起点章节 爬虫爬取小说_网络爬虫_07

我们可以看到页面上的小说名称我们也已经抓取到了。

3、获取小说详细信息:
我们点开页面上的其中一个小说链接:

Python爬虫爬起点章节 爬虫爬取小说_python_08

这里有小说的一些相关信息和小说章节目录的地址。
我们想要的数据首先是小说全部章节目录的地址,然后是小说类别、小说作者、小说状态、小说最后更新时间。
我们先看小说全部章节目录的地址。用F12,我们看到:

Python爬虫爬起点章节 爬虫爬取小说_scrapy_09

小说全部章节地址放在了“class”属性为“btnlinks”的“p”节点的第一个“a”节点中。
我们还是用scrapy shell测试一下我们写的xpath语句。
键入命令,进入shell界面
scrapy shell "http://www.23us.so/xiaoshuo/13007.html"
在shell中键入命令:
response.xpath('//p[@class="btnlinks"]/a[1]/@href').extract_first()

Python爬虫爬起点章节 爬虫爬取小说_网络爬虫_10

小说的章节目录页面我们已经截取下来了。

类似的还有小说类别、小说作者、小说状态、小说最后更新时间,命令分别是:
#小说类别
response.xpath('//table/tr[1]/td[1]/a/text()').extract_first()    
#小说作者
response.xpath('//table/tr[1]/td[2]/text()').extract_first()   
#小说状态
response.xpath('//table/tr[1]/td[3]/text()').extract_first()   
#小说最后更新时间
response.xpath('//table/tr[2]/td[3]/text()').extract_first()

Python爬虫爬起点章节 爬虫爬取小说_网络爬虫_11

4、获取小说全部章节:
我们点开“最新章节”,来到小说全部章节页面。

Python爬虫爬起点章节 爬虫爬取小说_python_12

我们如何获得这些链接呢?答案还是Xpath。
用F12看到,各章节地址和章节名称放在了一个“table”中:

Python爬虫爬起点章节 爬虫爬取小说_python_13


退出上次的scrapy shell ,分析 全部章节页面。

scrapy shell "http://www.23us.so/files/article/html/13/13007/index.html"

在shell中键入Xpath语句:

response.xpath('//table/tr/td/a/@href').extract()

Python爬虫爬起点章节 爬虫爬取小说_scrapy_14

同样还有各章节名称
response.xpath('//table/tr/td/a/text()').extract()

Python爬虫爬起点章节 爬虫爬取小说_python_15

5、爬取小说章节内容:

好了,小说各个章节地址我们截取下来了,接下来就是小说各个章节的内容。

我们用F12看到,章节内容放在了“id”属性为“contents”的“dd”节点中。

Python爬虫爬起点章节 爬虫爬取小说_scrapy_16

这里我们再用Xpath看一下,键入Xpath语句:
Response.xpath('//dd[@id="contents"]').extract()

Python爬虫爬起点章节 爬虫爬取小说_网络爬虫_17

我们看到,小说内容已经让我们截取到了!

二、编写爬虫:

整个流程上面已经介绍过了,还有一个非常重要的问题:

断点续传问题

我们知道,爬虫不可能一次将全部网站爬取下来,网站的数据量相当庞大,在短时间内不可能完成爬虫工作,在下一次启动爬虫时难道再将已经做过的工作再做一次?当然不行,这样的爬虫太不友好。那么我们如何来解决断点续传问题呢?

我这里的方法是,将已经爬取过的小说每一章的链接存入Mongodb数据库的一个集合中。在爬虫工作时首先检测,要爬取的章节链接是否在这个集合中:

如果在,说明这个章节已经爬取过,不需要再次爬取,跳过;

如果不在,说明这个章节没有爬取过,则爬取这个章节。爬取完成后,将这个章节链接存入集合中;

如此,我们就完美实现了断点续传问题,十分好用。

接下来贴出整个项目代码:

注释我写的相当详细,熟悉一下就可以看懂。

items.py

# -*- coding: utf-8 -*-

# Define here the models for your scraped items
#
# See documentation in:
# https://doc.scrapy.org/en/latest/topics/items.html

import scrapy


class DingdianxiaoshuoItem(scrapy.Item):
    # define the fields for your item here like:
    # name = scrapy.Field()
    #小说名字
    novel_name=scrapy.Field()
    #小说类别
    novel_family=scrapy.Field()
    #小说主页地址
    novel_url=scrapy.Field()
    #小说作者
    novel_author=scrapy.Field()
    #小说状态
    novel_status=scrapy.Field()
    #小说字数
    novel_number=scrapy.Field()
    #小说所有章节页面
    novel_all_section_url= scrapy.Field()
    #小说最后更新时间
    novel_updatetime=scrapy.Field()

    #存放小说的章节地址,程序中存放的是一个列表
    novel_section_urls=scrapy.Field()

    #存放小说的章节地址和小说章节名称的对应关系,程序中存储的是一个字典
    section_url_And_section_name=scrapy.Field()

dingdian.py

# -*- coding: utf-8 -*-
import scrapy
from scrapy import Selector
from dingdianxiaoshuo.items import DingdianxiaoshuoItem

class dingdian(scrapy.Spider):
    name="dingdian"
    allowed_domains=["23us.so"]
    start_urls = ['http://www.23us.so/top/allvisit_1.html']
    server_link='http://www.23us.so/top/allvisit_'
    link_last='.html'

    #从start_requests发送请求
    def start_requests(self):
        yield scrapy.Request(url = self.start_urls[0], callback = self.parse1)


    #获取总排行榜每个页面的链接
    def parse1(self, response):
        items=[]
        res = Selector(response)
        #获取总排行榜小说页码数
        max_num=res.xpath('//*[@id="pagestats"]/text()').extract_first()
        max_num=max_num.split('/')[1]
        print("总排行榜最大页面数为:"+max_num)
        #for i in max_num+1:
        for i in range(0,int(max_num)):
            #构造总排行榜中每个页面的链接
            page_url=self.server_link+str(i)+self.link_last
            yield scrapy.Request(url=page_url,meta={'items':items},callback=self.parse2)


    #访问总排行榜的每个页面
    def parse2(self,response):
        print(response.url)
        items=response.meta['items']
        res=Selector(response)
        #获得页面上所有小说主页链接地址
        novel_urls=res.xpath('//td/a[not(@title)]/@href').extract()
        #获得页面上所有小说的名称
        novel_names=res.xpath('//td/a[not(@title)]/text()').extract()

        page_novel_number=len(novel_urls)
        for index in range(page_novel_number):
            item=DingdianxiaoshuoItem()
            item['novel_name']=novel_names[index]
            item['novel_url'] =novel_urls[index]
            items.append(item)

        for item in items:
            #访问每个小说主页,传递novel_name
            yield scrapy.Request(url=item['novel_url'],meta = {'item':item},callback = self.parse3)

    #访问小说主页,继续完善item
    def parse3(self, response):
        #接收传递的item
        item=response.meta['item']
        #写入小说类别
        item['novel_family']=response.xpath('//table/tr[1]/td[1]/a/text()').extract_first()
        #写入小说作者
        item['novel_author']=response.xpath('//table/tr[1]/td[2]/text()').extract_first()
        #写入小说状态
        item['novel_status']=response.xpath('//table/tr[1]/td[3]/text()').extract_first()
        #写入小说最后更新时间
        item['novel_updatetime']=response.xpath('//table/tr[2]/td[3]/text()').extract_first()
        #写入小说全部章节页面
        item['novel_all_section_url']=response.xpath('//p[@class="btnlinks"]/a[1]/@href').extract_first()
        url=response.xpath('//p[@class="btnlinks"]/a[@class="read"]/@href').extract_first()
        #访问显示有全部章节地址的页面
        print("即将访问"+item['novel_name']+"全部章节地址")
        #yield item
        yield  scrapy.Request(url=url,meta={'item':item},callback=self.parse4)

    #将小说所有章节的地址和名称构造列表存入item
    def parse4(self, response):
        #print("这是parse4")
        #接收传递的item
        item=response.meta['item']
        #这里是一个列表,存放小说所有章节地址
        section_urls=response.xpath('//table/tr/td/a/@href').extract()
        #这里是一个列表,存放小说所有章节名称
        section_names=response.xpath('//table/tr/td/a/text()').extract()

        item["novel_section_urls"]=section_urls
        #计数器
        index=0
        #建立哈希表,存储章节地址和章节名称的对应关系
        section_url_And_section_name=dict(zip(section_urls,section_names))
        #将对应关系,写入item
        item["section_url_And_section_name"]=section_url_And_section_name


        yield item

settings.py

# -*- coding: utf-8 -*-

# Scrapy settings for dingdianxiaoshuo project
#
# For simplicity, this file contains only settings considered important or
# commonly used. You can find more settings consulting the documentation:
#
#     https://doc.scrapy.org/en/latest/topics/settings.html
#     https://doc.scrapy.org/en/latest/topics/downloader-middleware.html
#     https://doc.scrapy.org/en/latest/topics/spider-middleware.html

BOT_NAME = 'dingdianxiaoshuo'

SPIDER_MODULES = ['dingdianxiaoshuo.spiders']
NEWSPIDER_MODULE = 'dingdianxiaoshuo.spiders'


# Crawl responsibly by identifying yourself (and your website) on the user-agent
#USER_AGENT = 'dingdianxiaoshuo (+http://www.yourdomain.com)'

# Obey robots.txt rules
ROBOTSTXT_OBEY = False

# Configure maximum concurrent requests performed by Scrapy (default: 16)
#CONCURRENT_REQUESTS = 32

# Configure a delay for requests for the same website (default: 0)
# See https://doc.scrapy.org/en/latest/topics/settings.html#download-delay
# See also autothrottle settings and docs
DOWNLOAD_DELAY = 0.25


#CLOSESPIDER_TIMEOUT = 60 # 后结束爬虫


# The download delay setting will honor only one of:
#CONCURRENT_REQUESTS_PER_DOMAIN = 16
#CONCURRENT_REQUESTS_PER_IP = 16

# Disable cookies (enabled by default)
COOKIES_ENABLED = False

# Disable Telnet Console (enabled by default)
#TELNETCONSOLE_ENABLED = False

# Override the default request headers:
#DEFAULT_REQUEST_HEADERS = {
#   'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
#   'Accept-Language': 'en',
#}

# Enable or disable spider middlewares
# See https://doc.scrapy.org/en/latest/topics/spider-middleware.html
#SPIDER_MIDDLEWARES = {
#    'dingdianxiaoshuo.middlewares.DingdianxiaoshuoSpiderMiddleware': 543,
#}

# Enable or disable downloader middlewares
# See https://doc.scrapy.org/en/latest/topics/downloader-middleware.html
#DOWNLOADER_MIDDLEWARES = {
#    'dingdianxiaoshuo.middlewares.DingdianxiaoshuoDownloaderMiddleware': 543,
#}

# Enable or disable extensions
# See https://doc.scrapy.org/en/latest/topics/extensions.html
#EXTENSIONS = {
#    'scrapy.extensions.telnet.TelnetConsole': None,
#}

# Configure item pipelines
# See https://doc.scrapy.org/en/latest/topics/item-pipeline.html
ITEM_PIPELINES = {
    'dingdianxiaoshuo.pipelines.DingdianxiaoshuoPipeline': 300,
}

# Enable and configure the AutoThrottle extension (disabled by default)
# See https://doc.scrapy.org/en/latest/topics/autothrottle.html
#AUTOTHROTTLE_ENABLED = True
# The initial download delay
#AUTOTHROTTLE_START_DELAY = 5

pipeline.py

# -*- coding: utf-8 -*-

# Define your item pipelines here
#
# Don't forget to add your pipeline to the ITEM_PIPELINES setting
# See: https://doc.scrapy.org/en/latest/topics/item-pipeline.html

#因为爬取整个网站时间较长,这里为了实现断点续传,我们把每个小说下载完成的
#章节地址存入数据库一个单独的集合里,记录已完成抓取的小说章节

from pymongo import MongoClient
from urllib import request
from bs4 import BeautifulSoup

#在pipeline中我们将实现下载每个小说,存入MongoDB数据库

class DingdianxiaoshuoPipeline(object):
    def process_item(self, item, spider):
        #print("马衍硕")
        #如果获取章节链接进行如下操作
        if "novel_section_urls" in item:
            # 获取Mongodb链接
            client = MongoClient("mongodb://127.0.0.1:27017")
            #连接数据库
            db =client.dingdian
            #获取小说名称
            novel_name=item['novel_name']
            #根据小说名字,使用集合,没有则创建
            novel=db[novel_name]

            #使用记录已抓取网页的集合,没有则创建
            section_url_downloaded_collection=db.section_url_collection

            index=0
            print("正在下载:"+item["novel_name"])


            #根据小说每个章节的地址,下载小说各个章节
            for section_url in item['novel_section_urls']:

                #根据对应关系,找出章节名称
                section_name=item["section_url_And_section_name"][section_url]
                #如果将要下载的小说章节没有在section_url_collection集合中,也就是从未下载过,执行下载
                #否则跳过
                if  not section_url_downloaded_collection.find_one({"url":section_url}):
                    #使用urllib库获取网页HTML
                    response = request.Request(url=section_url)
                    download_response = request.urlopen(response)
                    download_html = download_response.read().decode('utf-8')
                    #利用BeautifulSoup对HTML进行处理,截取小说内容
                    soup_texts = BeautifulSoup(download_html, 'lxml')
                    content=soup_texts.find("dd",attrs={"id":"contents"}).getText()


                    #向Mongodb数据库插入下载完的小说章节内容
                    novel.insert({"novel_name": item['novel_name'], "novel_family": item['novel_family'],
                                  "novel_author":item['novel_author'], "novel_status":item['novel_status'],
                                  "section_name":section_name,
                                  "content": content})
                    index+=1
                    #下载完成,则将章节地址存入section_url_downloaded_collection集合
                    section_url_downloaded_collection.insert({"url":section_url})


        print("下载完成:"+item['novel_name'])
        return item
三、启动项目,查看运行结果:
程序编写完成后,我们进入项目所在目录,键入命令启动项目:
scrapy crawl dingdian

启动项目后,我们通过Mongodb可视化工具–RoBo看到,我们成功爬取了小说网站,接下来的问题交给时间。

Python爬虫爬起点章节 爬虫爬取小说_scrapy_18

Python爬虫爬起点章节 爬虫爬取小说_python_19

当想中断爬虫时,直接关掉控制台。下次开启爬虫时将不会重复上次的工作,这就是断点续传的美妙之处。(严格意义上不会在上次终止的地点开始爬取,但是不会重复已经爬取的工作)