python爬虫基本框架:爬虫调度器,URL管理器,HTML解析器,HTML下载器,数据存储器。
如下图:
各模块的功能分别如下:
爬虫调度器负责协调其他四个模块的工作;·URL管理器负责管理URL链接,维护已经爬取的URL集合和未爬取的URL集合,提供获取新URL链接的接口;HTML下载器负责从URL管理器中获取仍未爬取的URL,并且下载HTML(requests);HTML解析器负责解析已下载的HTML网页,并从中获取新的URL提供给URL管理器(bs4);数据存储器用于将HTML解析器解析出来的数据通过文件或者数据库的形式存储起来(codecs)。
比如爬取百度百科中和“网络爬虫”有关的100个词条,首先爬取“网络爬虫”这一词条的网页,然后在这个网页中爬取有关词条的链接,以此类推。这样的话,上述模块就分别如下:
数据存储器:
# -*- coding: utf-8 -*- import codecs from urllib import unquote class DataOutput(object): def __init__(self): self.datas = [] def store_data(self, data): if data is None: return self.datas.append(data) def output_html(self): fout = codecs.open('baike.html', 'w', encoding='utf-8') fout.write("<html>") fout.write("<body>") fout.write("<table>") for data in self.datas: #data['url']=unquote(data['url']) fout.write("<tr>") fout.write("<td>%s</td>" % data['url']) fout.write("<td>%s</td>" % data['title']) fout.write("<td>%s</td>" % data['summary']) fout.write("</tr>") self.datas.remove(data) fout.write("</table>") fout.write("</body>") fout.write("</html>") fout.close()
HTML解析器:
# -*- coding: utf-8 -*- import re import urlparse from bs4 import BeautifulSoup class HtmlParser(object): def parser(self, page_url, html_cont): ''' 用于解析网页内容,抽取URL和数据 :param page_url: 下载页面的URL :param html_cont: 下载的网页内容 :return:返回URL和数据 ''' if page_url is None or html_cont is None: return soup = BeautifulSoup(html_cont, 'html.parser', from_encoding='utf-8') new_urls = self._get_new_urls(page_url, soup) new_data = self._get_new_data(page_url, soup) return new_urls, new_data def _get_new_urls(self, page_url, soup): ''' 抽取新的URL集合 :param page_url: 下载页面的URL :param soup:soup :return: 返回新的URL集合 ''' new_urls = set() # 抽取符合要求的a标记 links = soup.find_all('a', href=re.compile(r'/item/\w+')) for link in links: # 提取href属性 new_url = link['href'] # 拼接成完整网址 new_full_url = urlparse.urljoin(page_url, new_url) new_urls.add(new_full_url) return new_urls def _get_new_data(self, page_url, soup): ''' 抽取有效数据 :param page_url:下载页面的URL :param soup: :return:返回有效数据 ''' data = {} data['url'] = page_url title = soup.find('dd', class_='lemmaWgt-lemmaTitle-title').find('h1') data['title'] = title.get_text() summary = soup.find('div', class_='lemma-summary') # 获取tag中包含的所有文本内容,包括子孙tag中的内容,并将结果作为Unicode字符串返回 data['summary'] = summary.get_text() return data
爬虫调度器:
# -*- coding: utf-8 -*- from DataOutput import DataOutput from HtmlDownloader import HtmlDownloader from HtmlParser import HtmlParser from UrlManager import UrlManager class SpiderMan(object): def __init__(self): self.manager = UrlManager() self.downloader = HtmlDownloader() self.parser = HtmlParser() self.output = DataOutput() def crawl(self, root_url): # 添加入口URL self.manager.add_new_url(root_url) # 判断url管理器中是否有新的url,同时判断抓取了多少个url while (self.manager.has_new_url() and self.manager.old_url_size() < 10): try: # 从URL管理器获取新的url new_url = self.manager.get_new_url() # HTML下载器下载网页 html = self.downloader.download(new_url) # HTML解析器抽取网页数据 new_urls, data = self.parser.parser(new_url, html) # 将抽取的url添加到URL管理器中 self.manager.add_new_urls(new_urls) # 数据存储器存储文件 self.output.store_data(data) print "已经抓取%s个链接" % self.manager.old_url_size() except Exception, e: print "crawl failed" # 数据存储器将文件输出成指定格式 self.output.output_html() if __name__ == "__main__": spider_man = SpiderMan() spider_man.crawl("https://baike.baidu.com/item/%E7%BD%91%E7%BB%9C%E7%88%AC%E8%99%AB")
URL管理器:
# -*- coding: utf-8 -*- class UrlManager(object): def __init__(self): self.new_urls = set() # 未爬取URL集合 self.old_urls = set() # 已爬取URL集合 def has_new_url(self): ''' 判断是否有未爬取的URL :return: ''' return self.new_url_size() != 0 def get_new_url(self): ''' 获取一个未爬取的URL :return: ''' new_url = self.new_urls.pop() self.old_urls.add(new_url) return new_url def add_new_url(self, url): ''' 将新的URL添加到未爬取的URL集合中 :param url:单个URL :return: ''' if url is None: return if url not in self.new_urls and url not in self.old_urls: self.new_urls.add(url) def add_new_urls(self, urls): ''' 将新的URL添加到未爬取的URL集合中 :param urls:url集合 :return: ''' if urls is None or len(urls) == 0: return for url in urls: self.add_new_url(url) def new_url_size(self): ''' 获取未爬取URL集合的大小 :return: ''' return len(self.new_urls) def old_url_size(self): ''' 获取已经爬取URL集合的大小 :return: ''' return len(self.old_urls)
HTML下载器:
# -*- coding: utf-8 -*- import requests class HtmlDownloader(object): def download(self, url): if url is None: return None user_agent = 'Mozilla/4.0 (compatible; MSIE 5.5; Windows NT)' headers = {'User-Agent': user_agent} r = requests.get(url, headers=headers) if r.status_code == 200: r.encoding = 'utf-8' return r.text return None
得到的结果部分截图如下:
很明显这其中需要urldecode,但是python并没有提供这个函数,尝试使用python提供的unquote并没有成功,具体原因以后再慢慢探索
加油( •̀ ω •́ )✧✴
PS:python版本:2.7
pycharm版本:community2020.2
很多很多相关内容摘自:《Python爬虫开发与项目实战》 — 范传辉 编著
在豆瓣阅读书店查看:https://read.douban.com/ebook/35093070/