爬虫 爬取糗事百科热门板块的信息

分析:

1、先查看返回的内容是否与显示的内容一样

2、再看页数的变化,每次会加一,总共13页,因此可以一次性构造地址列表

3、因此可直接结合 chrome插件 xpath helper 与 elemetns显示的内容进行定位要爬取的内容

 

用到的模块 requests+json+lxml+xpath

 下面是代码:

import requests
import json
from lxml import etree


class QiubaiSpider:
    def __init__(self):
        self.url_temp = "https://www.qiushibaike.com/8hr/page/{}/"
        self.headers = {
            "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/73.0.3683.103 Safari/537.36"}

    def get_url_list(self):  # 生成url_list
        return [self.url_temp.format(i) for i in range(1, 14)]

    def parse_url(self, url):  # 发送请求
        print(url)
        response = requests.get(url, headers=self.headers)
        return response.content

    def get_content_list(self,html_str):  # 提取数据
        html = etree.HTML(html_str)
        li_list = html.xpath("//div[@class='recommend-article']/ul/li")  # 分组
        content_list = []
        for li in li_list:
            item={}
            item["content"] = li.xpath("./div/a/text()") # 获取内容
            item['name'] = li.xpath("./div/div/a/span/text()") # 获取姓名
            item['name'] = item['name'][0] if len(item['name'])>0 else None
            item['content_img'] = li.xpath("./a/img/@src") # 获取图片
            item['content_img'] = 'https:'+item['content_img'][0] if len(item['content_img'])>0 else None
            item['auth_img'] = li.xpath(".//a[@class='recmd-user']/img/@src")  # 获取用户头像
            item['auth_img'] = 'https:'+item['auth_img'][0] if len(item['auth_img'])>0 else None
            item['recmd-num'] = li.xpath(".//div[@class='recmd-num']/span/text()")  # 获取点赞数和评论数
            content_list.append(item)
        return content_list

    def save_content_list(self,content_list):  # 保存数据
        with open('qiubai','a',encoding='utf-8')as f:
            for i in content_list:
                f.write(json.dumps(i,ensure_ascii=False,indent=4))
                f.write('\n')
    def run(self):  # 实现主要逻辑
        # 1 url_list
        url_list = self.get_url_list()
        # 2 遍历,发送请求,获取响应
        for url in url_list:
            html_str = self.parse_url(url)
            # 3 提取数据
            content_list = self.get_content_list(html_str)
            # 4 保存
            self.save_content_list(content_list)

if __name__ == '__main__':
    qiubai = QiubaiSpider()
    qiubai.run()

 

切换成多线程模式:

import requests
import json
from lxml import etree
import threading
from queue import Queue

class QiubaiSpider:
    def __init__(self):
        self.url_temp = "https://www.qiushibaike.com/8hr/page/{}/"
        self.headers = {
            "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/73.0.3683.103 Safari/537.36"}
        self.url_queue = Queue()  # url队列
        self.html_queue = Queue() # 响应队列
        self.content_queue = Queue() # 内容数据队列

    def get_url_list(self):  # 生成url_list
        # return [self.url_temp.format(i) for i in range(1, 14)]
        for i in range(1,14):
            self.url_queue.put(self.url_temp.format(i)) # 把url放入队列,每放入一个计数器加一

    def parse_url(self):  # 发送请求
        while True:
            url = self.url_queue.get()  # url从队列中取
            print(url)
            response = requests.get(url, headers=self.headers)
            self.html_queue.put(response.content) # 把响应结果放入响应队列
            self.url_queue.task_done()  # 一个响应完成。让计数器减一

    def get_content_list(self):  # 提取数据
        while True:
            html_str = self.html_queue.get()  # 从响应队列中提取响应的html文本
            html = etree.HTML(html_str)
            li_list = html.xpath("//div[@class='recommend-article']/ul/li")  # 分组
            content_list = []
            for li in li_list:
                item={}
                item["content"] = li.xpath("./div/a/text()") # 获取内容
                item['name'] = li.xpath("./div/div/a/span/text()") # 获取姓名
                item['name'] = item['name'][0] if len(item['name'])>0 else None
                item['content_img'] = li.xpath("./a/img/@src") # 获取图片
                item['content_img'] = 'https:'+item['content_img'][0] if len(item['content_img'])>0 else None
                item['auth_img'] = li.xpath(".//a[@class='recmd-user']/img/@src")  # 获取用户头像
                item['auth_img'] = 'https:'+item['auth_img'][0] if len(item['auth_img'])>0 else None
                item['recmd-num'] = li.xpath(".//div[@class='recmd-num']/span/text()")  # 获取点赞数和评论数
                content_list.append(item)
            self.content_queue.put(content_list)  # 把提取的数据放入内容数据队列
            self.html_queue.task_done()   # 从响应队列中每提取一个,计数器减一

    def save_content_list(self):  # 保存数据
        while True:
            content_list = self.content_queue.get() # 从内容数据队列中提取数据
            print(content_list)
            self.content_queue.task_done() # 内容数据队列中提取一个,计数器减一

    def run(self):  # 实现主要逻辑
        thread_list = []
        # 1 url_list
        t_url = threading.Thread(target=self.get_url_list)
        thread_list.append(t_url)
        # 2 遍历,发送请求,获取响应
        for i in range(5): # 设置5个线程
            t_parse = threading.Thread(target=self.parse_url)
            thread_list.append(t_parse)
        # 3 提取数据
        for i in range(3): # 设置3个线程
            t_html = threading.Thread(target=self.get_content_list)
            thread_list.append(t_html)
        # 4 保存
        t_save = threading.Thread(target=self.save_content_list)
        thread_list.append(t_save)
        for t in thread_list:
            t.setDaemon(True) # 把子线程设置为守护线程,该线程不重要,主线程结束,子线程结束
            t.start()
        for q in [self.url_queue,self.html_queue,self.content_queue]:
            q.join()     # 让主线程阻塞,子线程结束主线程才继续执行
        print('主线程结束')

if __name__ == '__main__':
    qiubai = QiubaiSpider()
    qiubai.run()

 

posted @ 2019-06-18 11:40  烛爻  阅读(196)  评论(0编辑  收藏  举报