专业爬虫框架 -- scrapy初识及基本应用

news2025/1/12 1:51:44

scrapy基本介绍

Scrapy一个开源和协作的框架,其最初是为了页面抓取 (更确切来说, 网络抓取 )所设计的,使用它可以以快速、简单、可扩展的方式从网站中提取所需的数据。

但目前Scrapy的用途十分广泛,可用于如数据挖掘、监测和自动化测试等领域,也可以应用在获取API所返回的数据(例如 Amazon Associates Web Services ) 或者通用的网络爬虫。Scrapy 是基于twisted框架开发而来,twisted是一个流行的事件驱动的python网络框架。

因此Scrapy使用了一种非阻塞(又名异步)的代码来实现并发。

 Scrapy架构

 百度上找到的Scrapy架构图:

1、引擎(Engine):
引擎负责控制系统所有组件之间的数据流,并在某些动作发生时触发事件。

有关详细信息,请参见上面的数据流部分。

------>>>

2、调度器(Scheduler):
用来接受引擎发过来的请求, 压入队列中, 并在引擎再次请求的时候返回. 可以想像成一个URL的优先级队列, 由它来决定下一个要抓取的网址是什么, 同时去除重复的网址

------>>>

3、下载器(Dowloader):
用于下载网页内容, 并将网页内容返回给Engine,下载器是建立在twisted这个高效的异步模型上的

------>>>

4、爬虫(Spiders):
SPIDERS是开发人员自定义的类,用来解析responses,并且提取items,或者发送新的请求

------>>>

5、项目管道(Item Pipelines):
在items被提取后负责处理它们,主要包括清理、验证、持久化(比如存到数据库)等操作
下载器中间件(Downloader Middlewares)位于Scrapy引擎和下载器之间,主要用来处理从Engine传到Downloader的请求request,已经从Downloader传到Engine的响应response。

------>>>

6、爬虫中间件(Spider Middlewares):
位于Engine和SPIDERS之间,主要工作是处理Spiders的输入(即responses)和输出(即requests)

 Scrapy安装

windows安装命令:pip3 install scrapy

依赖项安装:

pip3 install lxml

pip3 install whee

pip3 install pyopenssl

依赖项如果已经安装的可以跳过

官网链接:ghttps://docs.scrapy.org/en/latest/topics/commands.htmlscrapy官网链接:ghttps://docs.scrapy.org/en/latest/topics/commands.html

scrapy框架使用及命令详解

常用命令

查看帮助

 scrapy -h
 scrapy <command> -h

 有两种命令:其中Project-only必须切到项目文件夹下才能执行,而Global的命令则不需要

  Global commands

startproject #创建项目
genspider    #创建爬虫程序
settings     #如果是在项目目录下,则得到的是该项目的配置
runspider    #运行一个独立的python文件,不必创建项目
shell        #scrapy shell url地址  在交互式调试,如选择器规则正确与否
fetch        #独立于程单纯地爬取一个页面,可以拿到请求头
view         #下载完毕后直接弹出浏览器,以此可以分辨出哪些数据是ajax请求
version      #scrapy version 查看scrapy的版本,scrapy version -v查看scrapy依赖库的版本

Project-only commands

crawl        #运行爬虫,必须创建项目才行,确保配置文件中ROBOTSTXT_OBEY = False
check        #检测项目中有无语法错误
list         #列出项目中所包含的爬虫名
edit         #编辑器,一般不用
parse        #scrapy parse url地址 --callback 回调函数  #以此可以验证我们的回调函数是否正确
bench        #scrapy bentch压力测试

对于爬虫而言,我们需要关心及常用的命令就三个:

startproject创建项目、 genspider创建爬虫程序、crawl启动爬虫

创建项目

手动新建一个“day23”的文件夹,进入Teminal终端

scrapy startproject Newspro  #Newspro是项目名称

回车执行后就会自动帮我“day23”的文件夹下创建Newspro

ps:这里项目名称如果写成News pro,day23下父级目录会叫pro,然后是子级目录News

文件目录:

文件说明:

● scrapy.cfg:项目的主配置信息,用来部署scrapy时使用,爬虫相关的配置信息在settings.py文件中。

● items.py:设置数据存储模板,用于结构化数据,如:Django的Model
● pipelines:数据处理行为,如:一般结构化的数据持久化
● settings.py:配置文件,如:递归的层数、并发数,延迟下载等。强调:配置文件的选项必须大写否则视为无效,正确写法USER_AGENT='xxxx'
● spiders:爬虫目录,如:创建文件,编写爬虫规则

创建爬虫程序

上一步Teminal终端中创建完项目之后,已经提示需要先cd Newspro,开始创建爬虫程序:

cd Newspro  #进入项目文件夹
scrapy genspider wangyi news.163.com/  #创建爬虫程序1

即告诉Teminal终端:

我要用scrapy框架,创建(genspider)一个叫"wangyi"的爬虫程序,"news.163.com/"是要爬取的url,可以省略https://

可以看到在Spiders文件夹下,就自动帮我们生成了一个"wangyi.py"的文件,并且文件中自动写好了一个类,以及一些配置参数。

当然如果是老手也可以自己手动创建模块,但是小白的话更推荐用命令创建,不然模块中少写了参数,就会导致一些bug……

另外需要注意:里面的parse方法,parse这个方法名不能改,这是框架自带的回调函数

再创建一个环球新闻网的模块:也自动生成了一个"huanqiu.py"的模块

模块中的域名默认是按照http进行拼的,如果不对,也可以手动改成https

 scrapy genspider huanqiu huanqiu.com

在做爬虫的时候,我们可能不止爬取一个网站,规范是:

将它们全部放在'Spiders'文件夹下,每一个要爬取的网站建立单独一个模块,然后在模块里完善具体的爬虫逻辑和解析逻辑。

完整的创建项目 -> 创建爬虫程序代码如下:

scrapy startproject Newspro  #Newspro是项目名称
cd Newspro                   #进入项目文件夹
scrapy genspider wangyi news.163.com  #创建爬虫程序1
scrapy genspider huanqiu huanqiu.com  #创建爬虫程序2

Spider类详解

Spiders是定义如何抓取某个站点(或一组站点)的类,包括如何执行爬行(即跟随链接)以及如何从其页面中提取结构化数据(即抓取项目)。

换句话说,Spiders是您为特定站点(或者在某些情况下,一组站点)爬网和解析页面定义自定义行为的地方。 

=====================================================================

① 生成初始的Requests来爬取第一个URLS,并且标识一个回调函数;

第一个请求定义在start_requests()方法内,默认从start_urls列表中获得url地址来生成Request请求,默认的回调函数是parse方法。回调函数在下载完成返回response时自动触发

parse不能改名,必须叫parse

------->>>

② 在回调函数中,解析response并且返回值
 返回值可以4种:
          包含解析数据的字典
          Item对象
          新的Request对象(新的Requests也需要指定一个回调函数)
          或者是可迭代对象(包含Items或Request)

------->>>

③ 在回调函数中解析页面内容
通常使用Scrapy自带的Selectors,但很明显你也可以使用Beutifulsoup,lxml或其他你爱用啥用啥。

------->>>

④ 最后,针对返回的Items对象将会被持久化到数据库
   通过Item Pipeline组件存到数据库:https://docs.scrapy.org/en/latest/topics/item-pipeline.html#topics-item-pipeline)
   或者导出到不同的文件(通过Feed exports:https://docs.scrapy.org/en/latest/topics/feed-exports.html#topics-feed-exports)

启动爬虫程序

获取网易新闻的html信息,修改添加"wangyi.py"模块中的代码:

import scrapy

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址,可以放多个


    #回调函数,解析方法
    def parse(self, response):

        #windos系统记得写encoding="utf8",不然写入的会是一个空文件
        with open("new163..html","w",encoding="utf8") as f: 
            f.write(response.text)

启动爬虫程序方式1:Teminal在终端输入以下代码

ls
scrapy crawl wangyi   #crawl启动的意思,后面跟上要启动的模块名

回车之后,会输出很多日志,日志跑完,就会出现一个"new163"的html文件,说明执行成功

html文件可以直接用浏览器打开,就是网页新闻的页面

如果不想看到这一堆日志,可以在启动文件的时候,加上--nolog

一般刚开始调试的时候不建议关闭日志,否则哪里写错了,也看不到报错信息

scrapy crawl wangyi --nolog  #启动网易模块,且不显示日志

每次都得在终端输入命令还是有点麻烦,所以也有另一种启动方式

启动爬虫程序方式2:通过run来执行启动

scrapy框架没有自带的这个功能,我们可以创建一个py脚本文件

在项目的根目录下进行创建一个py文件,例如我的项目文件名叫"Newspro",就是在它下面创建

运行以下代码,也可以进行启动爬虫程序

from scrapy.cmdline import execute

#['scrapy', 'crawl', '文件名']
execute(['scrapy', 'crawl', 'wangyi'])

#需要关闭日志的话,加上"--nolog"即可
# execute(['scrapy', 'crawl', 'wangyi',"--nolog"])

项目使用--scrapy实战案例详解

基于前面创建的Newspro项目下的'wangyi.py'模块,批量爬取网易新闻首页板块tab的15个新闻分类里的所有新闻标题,如下截图圈出的板块:

-------------------------------------------------------------------------------------------------------------------------

当鼠标悬浮在tab标题上时,悬浮在不同标题会自动切换显示对应内容,说明此时非当前tab标题下的内容被隐藏了,浏览器并没有真正对服务器发起访问请求,比如说这个15个tab标题分类下,每个分类有100条新闻,那么总共就是1500条新闻,都在第一次的请求响应中。 

 通过之前爬取下来的html文件对比:

通过对比html文件就可以看出,响应信息只是放在不同位置 

⑴ 批量爬取网易新闻标题 

明确完整的爬取需求:

① tab板块中15个分类的所有新闻分类(即tab名称)

② 所有的新闻标题

③ 以及所有的新闻内容

=====================================================================

通过之前爬取下来的首页信息的html文件中可以看出,目前能拿到的是tab名称和新闻标题

新闻内容需要请求具体每个新闻的url,所以分三步来实现:

第一步:先来完成首页最容易获取的信息:解析新闻标题

第二步:解析tab分类名称

第三步:最后爬取具体的所有新闻内容,请求所有新闻的url进行获取

现在正式开始实现第一步:解析新闻标题

解析新闻标题 (错误示范)

先打开浏览器 -> f12进行元素xpath定位,先找到新闻标题对应的素: 

//div[@class="news_title"]/h3/a/text()

import scrapy

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址


    #回调函数,解析方法
    def parse(self, response):

        #获取新闻标题的列表:scrapy框架里也支持xpath语法
        news_title_list = response.xpath('//div[@class="news_title"]/h3/a')
        print("news_title_list::",news_title_list)
        print("news_title_list长度:",len(news_title_list)) #长度即新闻的个数

然后在bin模块的脚本文件中启动(关闭了日志):

bin文件执行完成之后,发现wangyi.py文件的输出并没有返回预期的信息

重新开启日志,再run一下:发现有报错

在上面的网页新闻f12元素中是能够找到新闻标题的,说明定义的xapth规则是没问题的

但是为什么解析不出来数据呢?这是一个坑,极大可能性是,在f12中我们看到的数据,是js渲染的,js把数据渲染到对应的标签里了

所以解决办法是:我们要先去看之前爬取下来的html文件,找到对应的hidden隐藏的属性值

解析新闻标题 (正确示范):重新修改代码

import scrapy

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址


    #回调函数,解析方法
    def parse(self, response):

        #获取新闻标题的列表:scrapy框架里也支持xpath语法
        news_title_list = response.xpath('//div[contains(@ne-if,"{{")]/div/a/text()')
        print("news_title_list::",news_title_list)
        print("news_title_list长度:",len(news_title_list)) #长度即新闻的个数

重新run一下bin文件:这次有返回信息,说明解析成功

但是框架自动帮我们用Selector封装了一个data,将新闻标题放在data中

我们只想要新闻标题,所以使用extract()进行提取:直接加在xapth后面就行

import scrapy

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址


    #回调函数,解析方法
    def parse(self, response):

        #获取新闻标题的列表:scrapy框架里也支持xpath语法
        news_title_list = response.xpath('//div[contains(@ne-if,"{{")]/div/a/text()').extract()
        print("news_title_list::",news_title_list)
        print("news_title_list长度:",len(news_title_list)) #长度即新闻的个数

重新run一下bin文件:这样就提取出了text值,即所有的新闻标题

extract()还有一个方法:extract_first(),值提取第一个

总结一下两个提取方法:

extract():提取Selector对象里的所有的data的text属性值
extract_first():提取Selector对象里的data的第一个text属性值,即索引为0的text

⑵ 爬取新闻分类+标题+链接

上面爬取新闻标题时,是对这个项目的初探

但是有个问题就是:只能获取到所有的新闻标题,没有新闻tab分类

=====================================================================

所以,我们重新转换一下思路:

❶ 先建立每个tab分类名称的关系映射字典 -> 在回调函数中写解析内容的代码:

这次直接提取网页新闻首页html文件中的分类title,也就是ne-if这个属性值

由于有15个分类板块,这里先取其中6个分类,进行循环15次,只返回在字典表中存在的

import scrapy

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址

    #建立一个tab分类名称关系映射字典:
    cate_mun_map = {
        "{{__i == 0}}" : "要闻" , "{{__i == 1}}": "上海",
        "{{__i == 3}}" : "国内" , "{{__i == 2}}": "国际",
        "{{__i == 4}}" : "独家" , "{{__i == 5}}": "军事",
        # "{{__i == 6}}" : "财经" , "{{__i == 7}}": "科技",
        # "{{__i == 8}}" : "体育" , "{{__i == 9}}": "娱乐",
        # "{{__i == 10}}": "时尚" , "{{__i == 11}}": "汽车",
        # "{{__i == 12}}": "房产" , "{{__i == 13}}": "航空",
        # "{{__i == 14}}": "健康"
    }


    #回调函数,解析方法
    def parse(self, response):

        #方式1:获取新闻标题的列表,scrapy框架里也支持xpath语法
        # news_title_list = response.xpath('//div[contains(@ne-if,"{{")]/div/a/text()').extract_first()
        # print("news_title_list::",news_title_list)
        # print("news_title_list长度:",len(news_title_list)) #长度即新闻的个数

        #方式2:获取标题和类别
        cate_mun__list = response.xpath('//*[contains(@ne-if,"{{")]/@ne-if').extract()
        print("cate_mun__list:",cate_mun__list)

        for cate_mun in cate_mun__list:
            #判断cate_mun是否在字典映射表中,只返回有的;循环15次,只拿符合条件的
            if cate_mun in self.cate_mun_map:
                #需要爬取的板块:
                cate_mun = self.cate_mun_map.get(cate_mun)
                print(cate_mun)

run以下bin文件:取到了6个分类

❷ 根据这6个分类,紧接着去循环爬取新闻标题和url:

这样就能将每个分类下的tab名称和新闻标题及url关联上

就相当于在上面代码中进行循环嵌套

import scrapy

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址

    #建立一个tab分类名称关系映射字典:
    cate_mun_map = {
        "{{__i == 0}}" : "要闻" , "{{__i == 1}}": "上海",
        "{{__i == 3}}" : "国内" , "{{__i == 2}}": "国际",
        "{{__i == 4}}" : "独家" , "{{__i == 5}}": "军事",
        # "{{__i == 6}}" : "财经" , "{{__i == 7}}": "科技",
        # "{{__i == 8}}" : "体育" , "{{__i == 9}}": "娱乐",
        # "{{__i == 10}}": "时尚" , "{{__i == 11}}": "汽车",
        # "{{__i == 12}}": "房产" , "{{__i == 13}}": "航空",
        # "{{__i == 14}}": "健康"
    }


    #回调函数,解析方法
    def parse(self, response):

        #方式1:获取新闻标题的列表,scrapy框架里也支持xpath语法
        # news_title_list = response.xpath('//div[contains(@ne-if,"{{")]/div/a/text()').extract_first()
        # print("news_title_list::",news_title_list)
        # print("news_title_list长度:",len(news_title_list)) #长度即新闻的个数

        #方式2:获取标题和类别
        cate_mun__list = response.xpath('//*[contains(@ne-if,"{{")]/@ne-if').extract()
        print("cate_mun__list:",cate_mun__list)

        for cate_mun in cate_mun__list:
            #判断cate_mun是否在字典映射表中,只返回有的;循环15次,只拿符合条件的
            if cate_mun in self.cate_mun_map:
                #需要爬取的板块:
                cate_title = self.cate_mun_map.get(cate_mun)
                print(cate_title)

                #爬取每一个板块的新闻标题:
                # response.xpath('//*[contains(@ne-if,"{{__i == 0}},{{__i == 1}}")/div/a')
                #不能写死i==0,==1,所以用他对应的cate_mun来做循环:
                news_selector_list = response.xpath(f'//*[contains(@ne-if,"{cate_mun}")]/div/a')
                for news_selector in news_selector_list:
                    news_title = news_selector.xpath("text()").extract_first()
                    news_link = news_selector.xpath("@href").extract_first()
                    print((news_title,news_link,cate_title))

再run一下bin文件:就得到了每个新闻分类下所有的新闻标题+每个新闻链接

⑶ 批量爬取新闻内容

根据上面取到的新闻分类、新闻标题、新闻链接

接下来还差最后一步:爬取到所有对应分类下的、所有新闻标题里的新闻内容

import scrapy
from scrapy.http import Request


class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址

    #建立一个tab分类名称关系映射字典:
    cate_mun_map = {
        "{{__i == 0}}" : "要闻" , "{{__i == 1}}": "上海",
        "{{__i == 3}}" : "国内" , "{{__i == 2}}": "国际",
        "{{__i == 4}}" : "独家" , "{{__i == 5}}": "军事",
        # "{{__i == 6}}" : "财经" , "{{__i == 7}}": "科技",
        # "{{__i == 8}}" : "体育" , "{{__i == 9}}": "娱乐",
        # "{{__i == 10}}": "时尚" , "{{__i == 11}}": "汽车",
        # "{{__i == 12}}": "房产" , "{{__i == 13}}": "航空",
        # "{{__i == 14}}": "健康"
    }


    #回调函数,解析方法
    def parse(self, response):

        #获取标题和类别
        cate_mun__list = response.xpath('//*[contains(@ne-if,"{{")]/@ne-if').extract()
        print("cate_mun__list:",cate_mun__list)

        for cate_mun in cate_mun__list:
            #判断cate_mun是否在字典映射表中,只返回有的;循环15次,只拿符合条件的
            if cate_mun in self.cate_mun_map:
                #需要爬取的板块:
                cate_title = self.cate_mun_map.get(cate_mun)
                print(cate_title)

                #爬取每一个板块的新闻标题:
                # response.xpath('//*[contains(@ne-if,"{{__i == 0}},{{__i == 1}}")/div/a')
                #不能写死i==0,==1,所以用他对应的cate_mun来做循环:
                news_selector_list = response.xpath(f'//*[contains(@ne-if,"{cate_mun}")]/div/a')
                for news_selector in news_selector_list:
                    news_title = news_selector.xpath("text()").extract_first()
                    news_link = news_selector.xpath("@href").extract_first()
                    print((news_title,news_link,cate_title))

                    '''
                     #根据源码照猫画虎:只要yield Request请求,就会自动帮我们将请求压缩到Scheduler、Dowloader
                     参数为url,去重,回调解析函数
                     这里的url是双层for循环下的6个新闻tab分类对应的700多个url
                     回调函数需要自定义,用于解析内容;回调函数是下面的parse_news_detail方法
                     注意:callback self的时候不要加括号
                    '''
                    yield Request(url=news_link,dont_filter=True,callback=self.parse_news_detail)

    #parse_news_detail方法,相当于是框架自动来完成700多次调用和响应
    def parse_news_detail(self, response):
        print("response::",response)

        #针对返回的response信息解析:
        #提取content的text内容
        content_list = response.xpath('//*[@id="content"]/div[@class="post_body"]/p/text()').extract()
        #拼接内容:
        content = "".join(content_list)
        print("content:",content)

run以下bin文件:但是content并没有被解析出来

原因是爬的过程,网易需要ua头,使用框架的好处就是,不再需要我们自己添加ua头

只需要在settings.py这个配置文件中,将17行放开,代码执行的过程中,框架就会自动拿到ua帮我放进去;

另外21行,是是否遵循机器人协议,默认为True,必要的时候可以改为False不遵循,但不建议改

然后重新run以下bin文件执行代码,还有两个问题

1、新闻分类的tab名称、和新闻标题及内容是分开的

2、新闻的content为空tab名称、和新闻标题及内容

先来解决第一个问题:tab名称、和新闻标题及内容把放在一起

在yield里面加上一个框架自带的meta字典,它作用是随着yield Request请求的发送,将各自的meta写进各自的response里面,这样就简单高效完成了放在一起的关联

import scrapy
from scrapy.http import Request

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址

    #建立一个tab分类名称关系映射字典:
    cate_mun_map = {
        "{{__i == 0}}" : "要闻" , "{{__i == 1}}": "上海",
        "{{__i == 3}}" : "国内" , "{{__i == 2}}": "国际",
        "{{__i == 4}}" : "独家" , "{{__i == 5}}": "军事",
        # "{{__i == 6}}" : "财经" , "{{__i == 7}}": "科技",
        # "{{__i == 8}}" : "体育" , "{{__i == 9}}": "娱乐",
        # "{{__i == 10}}": "时尚" , "{{__i == 11}}": "汽车",
        # "{{__i == 12}}": "房产" , "{{__i == 13}}": "航空",
        # "{{__i == 14}}": "健康"
    }


    #回调函数,解析方法
    def parse(self, response):

        #获取标题和类别
        cate_mun__list = response.xpath('//*[contains(@ne-if,"{{")]/@ne-if').extract()
        for cate_mun in cate_mun__list:
            #判断cate_mun是否在字典映射表中,只返回有的;循环15次,只拿符合条件的
            if cate_mun in self.cate_mun_map:
                #需要爬取的板块:
                cate_title = self.cate_mun_map.get(cate_mun)

                #爬取每一个板块的新闻标题:
                # response.xpath('//*[contains(@ne-if,"{{__i == 0}},{{__i == 1}}")/div/a')
                #不能写死i==0,==1,所以用他对应的cate_mun来做循环:
                news_selector_list = response.xpath(f'//*[contains(@ne-if,"{cate_mun}")]/div/a')
                for news_selector in news_selector_list:
                    news_title = news_selector.xpath("text()").extract_first()
                    news_link = news_selector.xpath("@href").extract_first()

                    '''
                     #根据源码照猫画虎:只要yield Request请求,就会自动帮我们将请求压缩到Scheduler、Dowloader
                     参数为url,去重,回调解析函数
                     这里的url是双层for循环下的6个新闻tab分类对应的700多个url
                     回调函数需要自定义,用于解析内容;回调函数是下面的parse_news_detail方法
                     注意:callback self的时候不要加括号
                    '''
                    yield Request(url=news_link,dont_filter=True,
                                  callback=self.parse_news_detail,
                                  meta={"news_title":news_title,"cate_title":cate_title})

    #parse_news_detail方法,相当于是框架自动来完成700多次调用和响应
    def parse_news_detail(self, response):
        news_title = response.meta.get("news_title")
        cate_title = response.meta.get("cate_title")

        #针对返回的response信息解析:
        #提取content的text内容
        content_list = response.xpath('//*[@id="content"]/div[@class="post_body"]/p/text()').extract()
        #拼接内容:
        content = "".join(content_list)
        print(cate_title,news_title,content)

Item及 PipeLine

Item(项目)

抓取的主要目标是从非结构化源(通常是网页)中提取结构化数据。

Scrapy蜘蛛可以像Python一样返回提取的数据。

虽然方便和熟悉,但很容易在字段名称中输入拼写错误或返回不一致的数据,尤其是在具有许多蜘蛛的较大项目中。

为了定义通用输出数据格式,Scrapy提供了Item类。

Item对象是用于收集抓取数据的简单容器。

它们提供类似字典的 API,并具有用于声明其可用字段的方便语法。

====================================================================

简单来说:

上面的实战案例中,我们初步完成了数据的爬取

接下来就是要进行数据清洗、和持久化存储到数据库

在'Newspro'的项目文件夹下,已经自动生成了一个'Item.py'的文件:

在Scrapy中,数据模型使用Item类来定义。在这个文件中,定义了一个名为NewsproItem的Item类,它继承自scrapy.Item类。

# define the fields for your item here like:
# name = scrapy.Field()

上面这两句话的意思是:

在这个Item类中,可以定义需要抓取的数据的字段。

每个字段都可以使用scrapy.Field()来定义,以便在抓取过程中存储相应的数据

上面"(3) 批量爬取新闻内容"最后的代码中"print(cate_title,news_title,content)",我们已经构建出来三个数据,由于没有一个统一的类来管理它们,所以输出打印结果看起来就很乱

所以我们在'Item.py'文件中使用scrapy.item来定义字段、管理它们:

import scrapy

class NewsItem(scrapy.Item):
    title = scrapy.Field()
    cata = scrapy.Field()
    content = scrapy.Field()

然后去"wangyi.py"文件中:进行封装item

import scrapy
from scrapy.http import Request
from NewsPro.items import NewsItem  #记得导入tiem;NewsPro是项目根目录

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址

    #建立一个tab分类名称关系映射字典:
    cate_mun_map = {
        "{{__i == 0}}" : "要闻" , "{{__i == 1}}": "上海",
        "{{__i == 3}}" : "国内" , "{{__i == 2}}": "国际",
        "{{__i == 4}}" : "独家" , "{{__i == 5}}": "军事",
        # "{{__i == 6}}" : "财经" , "{{__i == 7}}": "科技",
        # "{{__i == 8}}" : "体育" , "{{__i == 9}}": "娱乐",
        # "{{__i == 10}}": "时尚" , "{{__i == 11}}": "汽车",
        # "{{__i == 12}}": "房产" , "{{__i == 13}}": "航空",
        # "{{__i == 14}}": "健康"
    }


    #回调函数,解析方法
    def parse(self, response):

        #获取标题和类别
        cate_mun__list = response.xpath('//*[contains(@ne-if,"{{")]/@ne-if').extract()
        for cate_mun in cate_mun__list:
            #判断cate_mun是否在字典映射表中,只返回有的;循环15次,只拿符合条件的
            if cate_mun in self.cate_mun_map:
                #需要爬取的板块:
                cate_title = self.cate_mun_map.get(cate_mun)

                #爬取每一个板块的新闻标题:
                # response.xpath('//*[contains(@ne-if,"{{__i == 0}},{{__i == 1}}")/div/a')
                #不能写死i==0,==1,所以用他对应的cate_mun来做循环:
                news_selector_list = response.xpath(f'//*[contains(@ne-if,"{cate_mun}")]/div/a')
                for news_selector in news_selector_list:
                    news_title = news_selector.xpath("text()").extract_first()
                    news_link = news_selector.xpath("@href").extract_first()

                    '''
                     #根据源码照猫画虎:只要yield Request请求,就会自动帮我们将请求压缩到Scheduler、Dowloader
                     参数为url,去重,回调解析函数
                     这里的url是双层for循环下的6个新闻tab分类对应的700多个url
                     回调函数需要自定义,用于解析内容;回调函数是下面的parse_news_detail方法
                     注意:callback self的时候不要加括号
                    '''
                    yield Request(url=news_link,dont_filter=True,
                                  callback=self.parse_news_detail,
                                  meta={"news_title":news_title,"cate_title":cate_title})

    #parse_news_detail方法,相当于是框架自动来完成700多次调用和响应
    def parse_news_detail(self, response):
        news_title = response.meta.get("news_title")
        cate_title = response.meta.get("cate_title")

        #针对返回的response信息解析:
        #提取content的text内容
        content_list = response.xpath('//*[@id="content"]/div[@class="post_body"]/p/text()').extract()
        #拼接内容:
        content = "".join(content_list)
        print(cate_title,news_title,content)

        '''
        实例化封装item对象:
        目的1、统一数据
        目的2、方便item调度      
        '''
        newItem = NewsItem()
        NewsItem["title"] = news_title
        NewsItem["cate"] = cate_title
        NewsItem["content"] = content

        yield newItem

结合最上面的scrapy架构图来说说,为什么要封装item?

只要yield的是item对象,引擎就会把item交给pipelines,做数据清洗和存储(对应架构图的7到8);

如果yield的是Request对象,相当于是压到了Scheduler(第三步),重新请求Engine(第六步),重新响应解析;

----------------->>>>>

代码到这里,由于前面数据都已经准备好了,所以不需要在重新发起请求,直接return item,即构建了700多个新闻对应的item对象;

=====================================================================

所以总的来说:解析就两步

没解析完成,就yield Request

解析完成,拿到数据,就yield item对象

PipeLine

在一个项目被scpay抓取之后,它被发送到项目管道,该项目管道通过顺序执行的几个组件处理它。

每个项目管道组件(有时简称为“项目管道”)是一个实现简单方法的Python类。他们收到一个项目并对其执行操作,同时决定该项目是否应该继续通过管道或被丢弃并且不再处理。

项目管道的典型用途是:

> - cleansing HTML data:清理HTML数据
> - validating scraped data (checking that the items contain certain fields):验证爬取的数据
> - checking for duplicates (and dropping them):检查重复数据
> - storing the scraped item in a database:存储爬取到的数据

=====================================================================

每个项管道组件都是一个必须实现以下方法的Python类:

process_item(self,项目,蜘蛛)
为每个项目管道组件调用此方法:process_item() 

返回带数据的dict、一个Item (或任何后代类)对象,返回Twisted Deferred或引发 DropItem异常。丢弃的项目不再由其他管道组件处理。

-------------------------------------------------------------------------------------------------------------------------

此外,他们还可以实现以下方法:

open_spider(self,蜘蛛):打开蜘蛛时会调用此方法

close_spider(self,蜘蛛):当蜘蛛关闭时调用此方法。

from_crawler(cls,crawler ):如果存在,则调用此类方法以从a创建管道实例Crawler。它必须返回管道的新实例。Crawler对象提供对所有Scrapy核心组件的访问,如设置和信号; 它是管道访问它们并将其功能挂钩到Scrapy的一种方式。

上面提到,只要yield的是item对象,就会由item pipelines来接收

接下来,我们进入到项目创建时自动生成的"pipelines.py"文件中:

 直接添加打印item:

为了确保"pipelines.py"文件中的"process_item"方法会工作,需要去"settings.py"文件:

将66~68行的代码放开(默认是注释掉的),让它开启工作

Pipeline的作用:
"Newspro.pipelines.NewsproPipeline"可以构建多个,每个Pipeline都可以有不同的功能

比如:

写3个Pipeline,给每个Pipeline加上权重,执行的时候会依次按照从上到下的顺序:

第一个Pipeline是把数据存在文件里,执行完交给第二个Pipeline

第二个Pipeline是把数据存在mysql里,执行完交给第三个Pipeline

第三个Pipeline是把数据存在mongo里

多个Pipeline就是这样以此类推,到这里就是scrapy的最后一步闭环动作

然后在"wangyi.py"文件中,提取content的text内容时加上去空格(第60行):

import scrapy
from scrapy.http import Request
from Newspro.items import NewsItem

class WangyiSpider(scrapy.Spider):
    name = "wangyi"
    # allowed_domains = ["news.163.com"]   #限制爬的域一定要在这个url之下,所以先注释
    start_urls = ["http://news.163.com/"]  #爬虫起始地址

    #建立一个tab分类名称关系映射字典:
    cate_mun_map = {
        "{{__i == 0}}" : "要闻" , "{{__i == 1}}": "上海",
        "{{__i == 3}}" : "国内" , "{{__i == 2}}": "国际",
        "{{__i == 4}}" : "独家" , "{{__i == 5}}": "军事",
        # "{{__i == 6}}" : "财经" , "{{__i == 7}}": "科技",
        # "{{__i == 8}}" : "体育" , "{{__i == 9}}": "娱乐",
        # "{{__i == 10}}": "时尚" , "{{__i == 11}}": "汽车",
        # "{{__i == 12}}": "房产" , "{{__i == 13}}": "航空",
        # "{{__i == 14}}": "健康"
    }


    #回调函数,解析方法
    def parse(self, response):

        #获取标题和类别
        cate_mun__list = response.xpath('//*[contains(@ne-if,"{{")]/@ne-if').extract()
        for cate_mun in cate_mun__list:
            #判断cate_mun是否在字典映射表中,只返回有的;循环15次,只拿符合条件的
            if cate_mun in self.cate_mun_map:
                #需要爬取的板块:
                cate_title = self.cate_mun_map.get(cate_mun)

                #爬取每一个板块的新闻标题:
                # response.xpath('//*[contains(@ne-if,"{{__i == 0}},{{__i == 1}}")/div/a')
                #不能写死i==0,==1,所以用他对应的cate_mun来做循环:
                news_selector_list = response.xpath(f'//*[contains(@ne-if,"{cate_mun}")]/div/a')
                for news_selector in news_selector_list:
                    news_title = news_selector.xpath("text()").extract_first()
                    news_link = news_selector.xpath("@href").extract_first()

                    '''
                     #根据源码照猫画虎:只要yield Request请求,就会自动帮我们将请求压缩到Scheduler、Dowloader
                     参数为url,去重,回调解析函数
                     这里的url是双层for循环下的6个新闻tab分类对应的700多个url
                     回调函数需要自定义,用于解析内容;回调函数是下面的parse_news_detail方法
                     注意:callback self的时候不要加括号
                    '''
                    yield Request(url=news_link,dont_filter=True,
                                  callback=self.parse_news_detail,
                                  meta={"news_title":news_title,"cate_title":cate_title})

    #parse_news_detail方法,相当于是框架自动来完成700多次调用和响应
    def parse_news_detail(self, response):
        news_title = response.meta.get("news_title")
        cate_title = response.meta.get("cate_title")

        #针对返回的response信息解析:
        #提取content的text内容
        content_list = response.xpath('//*[@id="content"]/div[@class="post_body"]/p/text()').extract()
        #拼接内容、并去除空格
        content = "".join([i.strip() for i in content_list])
        print(cate_title,news_title,content)

        '''
        实例化封装item对象:
        目的1、统一数据
        目的2、方便item调度      
        '''
        newItem = NewsItem()
        newItem["title"] = news_title
        newItem["cate"] = cate_title
        newItem["content"] = content

        yield newItem

重新run一下bin文件:

cate(tab分类标题)、title(新闻标题)、content(新闻内容)已经放在一起了

但是仍有一些content内容没有解析到的,这是前面遗留的第二个问题("⑶ 批量爬取新闻内容"这里),是由于"news_selector_list"这里xapth定义的解析规则不能适用所有的新闻分类导致的

所以这里我再做下处理,先把content为空的过滤掉(在"process_item"文件里处理)

可以导入from scrapy.extensions import DropItem这个类,用来丢弃item

from scrapy.extensions import DropItem #丢弃item

class NewsproPipeline:
    def process_item(self, item, spider):

        #加个判断条件,content为空的不返回
        if not item["content"]:
           DropItem("content不能为空,丢弃!")
        else:
            print("item:::", item)
            return item #一定要记得加return进行传递

再重新run一下bin文件:在控制台搜索一下为空的content,显示0个,说明已经过滤成功了

到这里,就已经使用scrapy框架完成了整个爬虫过程:请求url -> 解析数据  -> 清洗数据

下一章接着一起来学习scrapy框架的进一步学习哦~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1281973.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

HarmonyOS4.0之安装DevEco Studio开发工具

第一步 打开网址&#xff1a;https://developer.huawei.com/consumer/cn/ 点击后是这样的界面 第二步 鼠标移入到开发点击DevEco Studio 第三步 我们往下滑动找到以下界面 我们根据自己的需要点击下载图标 这里演示Window系统 下载好后解压文件 我们解压文件后 第四步…

VS2022配置WinPcap开发

winpcap 官网&#xff1a;http://www.winpcap.org/ 1.首先下载安装 winpcap.exe&#xff0c;http://www.winpcap.org/install/default.htm 目的是安装相关驱动和 dll&#xff0c;安装完成之后基于 winpcap 的应用程序才能够正常运行。 2.下载 winpcap 的开发包&#xff0c;头文…

34、AD/DA

AD/DA介绍 AD&#xff08;Analog to Digital&#xff09;&#xff1a;模拟-数字转换&#xff0c;将模拟信号转换为计算机可操作的数字信号 DA&#xff08;Digital to Analog&#xff09;&#xff1a;数字-模拟转换&#xff0c;将计算机输出的数字信号转换为模拟信号 AD/DA转换…

离线数仓构建案例一

数据采集 日志数据&#xff08;文件&#xff09;到Kafka 自己写个程序模拟一些用户的行为数据&#xff0c;这些数据存在一个文件夹中。 接着使用flume监控采集这些文件&#xff0c;然后发送给kafka中待消费。 1、flume采集配置文件 监控文件将数据发给kafka的flume配置文件…

python scipy.cluster.hierarchy.dendrogram学习详记——(待完善)

1.Python scipy.cluster.hierarchy.dendrogram用法及代码示例 2.python dendrogram_Python中的凝聚层次聚类示例

微信小程序在线客服 全端通吃版+PC官网客服+H5网站客服+微信公众号客服 附带完整的搭建教程

随着互联网的快速发展&#xff0c;在线客服系统已经成为企业与用户沟通的重要桥梁。然而&#xff0c;许多企业在构建自己的在线客服系统时&#xff0c;往往面临多种平台、多端口的困扰&#xff0c;如何实现全端通吃的客服系统成为一项迫切的需求。为此&#xff0c;我们推出了一…

android.view.WindowLeaked解决方法

问题 我在使用WindowManager添加一个button&#xff0c; windowManager.addView(button,layoutParams);然后关闭当前的这个Activity的时候遇到了WindowLeak这个问题&#xff0c;也就是所谓的窗体泄露。 原因 主要原因是因为android只允许在UI主线程操作&#xff0c;我在使用W…

geoserver维度time

postgis创建date类型的字段 写入测试数据&#xff0c;对应flag&#xff0c;flag有不同的样式&#xff0c;这样方便观测 geoserver发布图层的时候设置“维度”启用 测试&#xff0c;设置了根据flag展示不同的颜色

本地源文件-丰富的图表-

D:\FineReport_11.0\webapps\webroot\WEB-INF\reportlets\demo\basic 图表类型:http://localhost:8075/webroot/help/demo.html 可视化图表&#xff0c;丰富的图表:help/demo.html http://localhost:8075/webroot/decision#management/directory 参数查询/条件查询与图…

使用Redis构建简易社交网站(1)-创建用户与动态界面

目的 本文目的&#xff1a;实现简易社交网站中创建新用户和创建新动态功能。&#xff08;完整代码附在文章末尾&#xff09; 相关知识 本文将教会你掌握&#xff1a;1.redis基本命令&#xff0c;2.python基本命令。 redis基本命令 hget&#xff1a;从哈希中获取指定域的值…

h5进行svga动画礼物特效播放的代码实现队列按顺序播放

需求描述&#xff1a; 在直播场景中&#xff0c;有很多的礼物特效动画&#xff0c;如采用Svga动画的播放方案&#xff0c;则会遇到以下问题&#xff1b; 1.svga文件的预加载&#xff0c; 2.动画的顺序播放队列。即前一个动画播放完了&#xff0c;才会播放下一个动画。 1.svg…

沉浸式观影怎么能少得了投影仪?极米轻薄投影极米Z7X了解一下

近段时间&#xff0c;各个平台好剧不断&#xff0c;《以爱为营》《宁安如梦》《乐源游》《无所畏惧》等优质好剧陆续开播&#xff0c;让剧迷们直呼看不过来。优质好剧已经开场&#xff0c;看好剧的装备当然也不能落下。现如今&#xff0c;大屏追剧已成潮流,极米Z7X陪大家一起开…

【笔记】Clion 中运行 C/C++11 之 CMakeLists.txt 的配置

该文章记录第一次使用 Clion 时&#xff0c;对 CMakeLists 的配置&#xff0c;使其能够运行 C/C11 的代码。 一. CMakeLists.txt 的配置 1、首先我们在需要新建一个项目 2、填写新建项目相关的信息 3、修改 CMakeLists.txt 文件内容 替换文本&#xff1a; # 使用此 CMakeLis…

两道面试题秒杀你的C++基础!

大家好&#xff0c;我是光城&#xff0c;今天发两个非常重要的面试题&#xff0c;可以留言区说出你的答案&#xff0c;这两个题目都比较重要&#xff0c;看你能答对不&#xff1f; 1.C中初始化变量有几种方式&#xff0c;各自有什么区别&#xff1f; 或者说Initialization分为哪…

数学建模-数据新动能驱动中国经济增长的统计研究-基于数字产业化和产业数字化的经济贡献测度

数据新动能驱动中国经济增长的统计研究-基于数字产业化和产业数字化的经济贡献测度 整体求解过程概述(摘要) 伴随着数据要素化进程的不断加深&#xff0c;对于数据如何作用于经济发展&#xff0c;数据与其他要素结合产生的动能应该如何测度的研究愈发重要。本文将数据新动能分…

MySQL性能调优-1-实际优化案例

关于SQL优化的思路&#xff0c;一般都是使用执行计划看看是否用到了索引&#xff0c;主要可能有两大类情况&#xff1a; 对业务字段建立了二级联合索引&#xff0c;但是MySQL错误地觉得走主键聚族索引全表扫描效率更高&#xff0c;而没有走二级索引 走二级索引&#xff0c;但…

LLM | 一文了解大语言模型中的参数高效微调(PEFT)

Parameter Efficient Fine Tuning(PEFT)也就是参数高效微调&#xff0c;是一种用于微调大型语言模型 &#xff08;LLM&#xff09; 的方法&#xff0c;与传统方法相比&#xff0c;它有效地降低了计算和内存需求。PEFT仅对模型参数的一小部分进行微调&#xff0c;同时冻结大部分…

9款热门API接口分享,值得收藏!

电商API接口 干货分享 开始 “ API是什么&#xff1f; API的主要目的是提供应用程序与开发人员以访问一组例程的能力&#xff0c;而又无需访问源码&#xff0c;或理解内部工作机制的细节。提供API所定义的功能的软件称作此API的实现。API是一种接口&#xff0c;故而是一种抽象…

使用PCSS实现的实时阴影效果

PCSS的技术可以使得阴影呈现出近硬远软的效果&#xff0c;并且能够实时实现。 其核心理念是通过模拟光源的面积来产生更自然、更柔和的阴影边缘。 具体步骤&#xff1a; 1、生成shadowmap 2、在进行阴影的比较时候进行平均&#xff0c;并非之前的shadow map 或者之后完全的阴影…

Xshell全局去除提示音

使用Xshell的时候经常会按TAB或者一些操作指令的时候的时候听到提示音&#xff0c;非常的烦 通常来说在Xshell中可以单独修改每一个会话的属性&#xff0c;将提示音关闭&#xff0c;但是新增的会话依然带有提示音&#xff0c;还得一个个的关闭&#xff0c;非常麻烦&#xff0c;…