简单爬虫架构的实现

在我的上一篇文章里简单介绍了一下最简单的爬虫架构:《浅谈简单爬虫架构》
如下图所示

简单爬虫架构

框架

mySpider
├─ spiderMain.py #爬虫调度端
├─ urlManager.py #URL管理器
├─ htmlDownloader.py #网页下载器
└─ htmlParser.py #网页解析器

此篇以爬取廖雪峰的官方网站中的python教程为例

不过廖老师的网站对爬虫进行了过滤,建议举一反三,尝试爬取其他网站

现在的网络爬虫越来越多,有很多爬虫都是初学者写的,和搜索引擎的爬虫不一样,他们不懂如何控制速度,结果往往大量消耗服务器资源,导致带宽白白浪费了。
其实Nginx可以非常容易地根据User-Agent过滤请求,我们只需要在需要URL入口位置通过一个简单的正则表达式就可以过滤不符合要求的爬虫请求:
...
location / {
if ($http_user_agent ~* "python|curl|java|wget|httpclient|okhttp") {
return 503;
}
# 正常处理
...
}
...
变量$http_user_agent是一个可以直接在location中引用的Nginx变量。~*表示不区分大小写的正则匹配,通过python就可以过滤掉80%的Python爬虫

爬虫调度端

爬虫调度端的核心代码实现:

while urlManager.hasUrl(): #询问url管理器是否有待爬url
        newurl = urlManager.getUrl() #获取待爬url
        html = htmlDownloader.download(newurl,headers) #下载页面
        (title,content),urls = htmlParser.parser(html) #从页面中解析出内容和新的url
  
        #此处可以处理爬下来的文件,比如储存到本地,我仅打印标题以测试
        print(title)

        urlManager.addUrls(urls)  #将新的url加入URL管理器

其中的headers是请求头,用于模拟浏览器的行为
可以使用chrome浏览器的开发者工具找到
网页右击 检查->network
刷新页面


右下角Request Headers

完整代码:

import requests
from urlManager import UrlManager #url管理器
from htmlDownloader import HtmlDownloader #页面下载器
from htmlParser import HtmlParser #网页解析器
#请求头
headers={'Accept':'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8','Accept-Encoding':'gzip, deflate, br','Accept-Language':'zh-CN,zh;q=0.9,en;q=0.8','Cache-Control':'max-age=0','Connection':'keep-alive','Cookie':'Hm_lvt_2efddd14a5f2b304677462d06fb4f964=1516595211; atsp=1516853801496_1516853801154; Hm_lpvt_2efddd14a5f2b304677462d06fb4f964=1516853818','Host':'www.liaoxuefeng.com','Referer':'https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0014318447437605e90206e261744c08630a836851f5183000','Upgrade-Insecure-Requests':'1','User-Agent':'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.132 Safari/537.36'}

if __name__ == '__main__':
    print('爬虫开始...')
    urlManager = UrlManager()
    htmlDownloader = HtmlDownloader()
    htmlParser = HtmlParser()
    urlManager.addUrl('https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000')
    while urlManager.hasUrl():
        newurl = urlManager.getUrl()
        if newurl != "":
            html = htmlDownloader.download(newurl,headers)
            (title,content),urls = htmlParser.parser(html)
            print(title)
        #文件处理储存
            urlManager.addUrls(urls)

URL管理器

url管理器可以使用queue或set甚至list来实现,如果需要按顺序爬取,可以使用队列来实现,即先加入url管理器的url先爬取,但是需要注意的是使用队列则需要检查待加入的url是否已经在队列中了。
而如果对顺序没有特别的要求,使用set更为简便,可以直接加入待爬url,因为重复的元素只会在set中出现一次。

但无论使用queue、set还是list实现,都需要检查待加的url是否已经爬过了,以此避免重复爬取甚至循环爬取

完整代码:

class UrlManager(object):

    def __init__(self):
        self._newUrls = set([]) #set newUrls中储存未访问的url 
        self._oldUrls = set([]) #set oldUrls中储存已访问的url

    def addUrl(self,url): #添加url
        if isinstance(url,str) and url not in self._oldUrls: 
            self._newUrls.add(url)

    def hasUrl(self): #是否还有未访问的url
        return len(self._newUrls) > 0 

    def getUrl(self): #该函数返回url视为已访问
        if  not self._newUrls: #不存在新的url
            return ""
        url = self._newUrls.pop()
        self._oldUrls.add(url)
        return url

    def addUrls(self,urls):
        if isinstance(urls,list):
            for url in urls:
                if url not in self._oldUrls:
                    self._newUrls.add(url)

网页下载器

网页下载器十分简单,在此使用了request模块
详情可以查看request模块文档

完整代码:

import requests

class HtmlDownloader(object):
    
    def download(self,url,headers=""): #url:待爬链接 headers:请求头  return value:html文本
        if url is None or not isinstance(url,str):
            #获取失败错误处理
            return None
        r = requests.get(url,headers=headers)
        return r.text

网页解析器

网页解析器使用了BeautifulSoup模块,非常方便快捷
具体参考BeautifulSoup文档
像我就记不太住,都是随用随查的,所以这是最费时间的一部分,也很容易出错(逃
这里需要我们自己分析页面来解析

完整代码:

from bs4 import BeautifulSoup

baseUrl = 'https://www.liaoxuefeng.com'

class HtmlParser(object):
    def parser(self,text): #text: html文本  return value: (需要的数据,list[需要的url])
        soup = BeautifulSoup(text,'lxml')
        #获取标题
        title = soup.h4
        #print(soup.select(".x-main-content"))
        if len(soup.select(".x-main-content")) >0:
        #   print("True")
            content = soup.select(".x-main-content")[0].get_text()
        else:
        #   print("False")
            content=""
        urls = []
        for a in soup.find_all('a',"x-wiki-index-item"):
            urls.append(baseUrl+a.get('href'))  
        return ((title,content),urls)

价值数据

做到这里,我们就已经可以获取到我们想要的价值数据了。
在这里我仅打印标题以测试,实际上我们可以对数据做更多的处理,而这就需要我们继续深入学习了。共勉!

爬虫结果
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • 爬虫文章 in 简书程序员专题: like:128-Python 爬取落网音乐 like:127-【图文详解】py...
    喜欢吃栗子阅读 21,631评论 4 412
  • Python开发简单爬虫(Python2.X版本,Eclipse工具) 一、爬虫介绍 爬虫调度端:启动、停止爬虫,...
    凛0_0阅读 2,110评论 2 10
  • 1 前言 作为一名合格的数据分析师,其完整的技术知识体系必须贯穿数据获取、数据存储、数据提取、数据分析、数据挖掘、...
    whenif阅读 17,988评论 45 523
  • 14天训练营结束了,感觉有点儿不自在。 训练营期间,睁开眼的第一件事就是看看今天的晨读推送了没有,如果看到有推送,...
    小沙弥_383c阅读 171评论 0 1
  • 不久,村里就接到了上面发下来的通知,说红旗村的土地将被国家占用,修建一个新县城,而村里人不仅将得到安置费,...
    宫长青筝阅读 106评论 0 0