新手向爬虫(四)Python爬取英文新书下载链接

该文是Python相关英文新书下载链接一文的代码实现部分,虽然有作者做过,但我还是自己重新用Scrapy实现了下,思路大致都差不多。

  • 与前面的 新手向爬虫(三)别人的爬虫在干啥绘制你的简书曲线 相比其实差别不大,多的地方在于

    1. 使用scrapy.Request函数的meta参数来在两次请求的响应中传值
    2. 对于动态js页面的处理:此次,网页是在加载之后自动执行js代码来修改html元素的属性,添加链接。首先我想到的是按着js代码来改出相应的python代码,用python代码的运算得到最终想要获取的值,虽然速度会快,但是比较繁琐,而且不够通用,也有可能有潜藏的问题,于是还是选择了最通用的selenium来控制浏览器加载页面,再获取自动运行js后的页面,然后利用正则分析页面,这样稍微慢一些,但是比较比较便捷稳定。
  • 此外,此次爬取的网盘它的资源是放在不同的服务器上的,对应的是wwwxy.zippyshare.com,xy是数字来区别不同的服务器,所以在页面上获取到相对路由后要加上正确的服务器域名;还有就是这个网站的网盘资源链接具有时效性,过一段时间后它会重置资源的获取路径,而原来的资源链接会重新路由到下载界面;关于后处理中的迅雷下载,我采用了将所有链接放在一个txt文件里的方法,复制所有链接时迅雷会侦听到并询问下载,同时也生成了另一篇文章中的markdown文件方便挑选。另外,还有个小bug就是scrapy会打开两个浏览器窗口,但是只用一个,不影响功能,暂时没管,有朋友赐教的话,非常感谢。


Scrapy爬虫

我命名文件为book1.py,使用scrapy runspider book1.py -o 2.json运行。
用的是selenium操作Chrome浏览器,Chromedriver下载,使用时要指定该可执行文件的位置。
修改第7行的search_what获取其它资源的所有相关书籍链接,这里我选的是python。

# -*- coding: utf-8 -*-
import scrapy

from selenium import webdriver
import re

search_what = 'python'
browser = webdriver.Chrome(r'E:\python\Scrapy2\book\book\spiders\chromedriver') # 指定Chromedriver.exe的位置

class Book1Spider(scrapy.Spider):
    name = "book1"
    allowed_domains = ["foxebook.net", "zippyshare.com"]
    
    down_base_url = ".zippyshare.com" # wwwxy,xy为数字代表不同服务器
    search_url = "http://www.foxebook.net/search/{}/page/{}"
    book_base_url = "http://www.foxebook.net"
    wd = browser # 获得浏览器对象
    reg = re.compile('<a id="dlbutton" href="(.*)">')
    
    headers = {
        "User-Agent":"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.71 Safari/537.36",
    }
    
    num = 0 # 页数

    def start_requests(self): # 默认的开始函数,用于提供要爬取的链接
        yield scrapy.Request(self.search_url.format(search_what, 0),
                             headers = self.headers,
                             callback = self.first_parse)
                
    def first_parse(self, response):
        max_page = response.css('.next a::attr("href")').extract_first().split('/')[-2]
        print(int(max_page))
        while self.num < int(max_page):
            self.num += 1
            yield scrapy.Request(self.search_url.format(search_what, self.num),
                                 headers = self.headers,
                                 callback = self.list_parse)
                             
    def list_parse(self, response):
        title = response.css('h3 a::text').extract()
        img = response.css('.img-responsive::attr("src")').extract()
        download = response.css(' .btn-info::attr("href")').extract()
        info =  response.css('.info~ .info+ .info i::text').extract()
        for t, im, d, inf in zip(title, img, download, info):
            meta = {}
            # 使用 meta参数 传值
            meta['title'] = t
            meta['img'] = im
            meta['info'] = inf
            yield scrapy.Request(self.book_base_url + d,
                                 meta = meta,
                                 headers = self.headers,
                                 callback = self.book_parse)
                                 
    def book_parse(self, response):                                
        book_url_list = response.css('.table-hover a::attr("href")').extract()
        for i in book_url_list:
            book_url = 'http' + i.split('http')[-1]
            if 'zip' in book_url: # 网盘名包含zip,以此做判断,其它网盘效果一般,不采用
                self.wd.get(book_url) # 在浏览器中打开,通过self.wd.page_source获取浏览器里的响应内容
                try:
                    down_url = book_url.split('.')[0] + self.down_base_url + re.findall(self.reg, self.wd.page_source)[0]
                except:
                    break
                yield {
                    'title':response.meta['title'], 
                    'img':response.meta['img'], 
                    'info':response.meta['info'],
                    'down_url':down_url, 
                    'web_pan':book_url,
                }
            else:
                pass

# Debug
                # from scrapy.shell import inspect_response
                # inspect_response(response, self)
# scrapy runspider book1.py -o 2.json

数据后处理

# -*- coding: utf-8 -*-
import json

with open('2.json','r') as f:
    data = json.load(f)
    data = sorted(data, key =lambda x: x['info'][2:12], reverse=True)
    # 以出版日期对文章进行排序
    
    with open("1.txt", 'w') as t: 
    # 生成便于迅雷下载的链接文件
    # 先打开迅雷,再复制该文件内容,迅雷会自动弹出询问是否下载所有链接
        for i in data:
            t.write(i['down_url']+'\n')
            
    with open('1.md', 'w') as m: # 书籍有可能重复,但是链接是不同的,指向不同服务器
        m.write('### 英文Python电子书籍下载链接:\n')
        for i in data:
            m.write('- **[{title}]({down_url})** \n *{info}*\n ![](http:{img}) \n --- \n'.format(**i))
            # **i 表示解包字典

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

推荐阅读更多精彩内容

  • 爬虫文章 in 简书程序员专题: like:128 - Python 爬取落网音乐 like:127 - 【图文详...
    treelake阅读 29,544评论 33 638
  • Python开发简单爬虫(Python2.X版本,Eclipse工具) 一、爬虫介绍 爬虫调度端:启动、停止爬虫,...
    凛0_0阅读 2,133评论 2 10
  • python中一切皆为对象 其实面向对象没什么高大上的东西,只不过把我们平时对于事物的描述和动作系统的总结成了一...
    Customer_阅读 756评论 0 0
  • 一、 “你跟我说什么八荣八耻!” “那是在教你,该披什么皮做人。” ​​​ 二、 这世上,本就不是事事都能苛责。 ​​
    宜尔er阅读 1,008评论 0 0