摘要:返回每個章節(jié)的集合。爬取完成,共計用了分鐘左右。判斷使用還是使用變量來存放未變化的,若下次循環(huán)與相等,說明此次請求沒有成功,,因為某些頁面本身存在錯誤沒有數據,則需要跳過。
一,獲取免費ip
關于免費ip,我選擇的是站大爺。因為免費ip的壽命很短,所以盡量要使用實時的ip,這里我專門使用getip.py來獲取免費ip,代碼會爬取最新的三十個ip,并以字典的形式返回兩種,如{’http‘:’ip‘},{’https‘:’ip‘}
?。。。。?!這里是另寫了一個py文件,后續(xù)正式寫爬蟲的時候會調用。
import requestsfrom lxml import etreefrom time import sleepdef getip(): base_url = "https://www.zdaye.com" url = "https://www.zdaye.com/dayProxy.html" headers = { "User-Agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36" } res = requests.get(url, headers=headers) res.encoding = "utf-8" dom = etree.HTML(res.text) sub_urls = dom.xpath("http://h3[@class ="thread_title"]/a/@href") sub_pages =[] for sub_url in sub_urls: for i in range(1, 11): sub_page = (base_url + sub_url).rstrip(".html") + "/" + str(i) + ".html" sub_pages.append(sub_page) http_list = [] https_list = [] for sub in sub_pages[:3]: sub_res = requests.get(sub, headers=headers) sub_res.encoding = "utf-8" sub_dom = etree.HTML(sub_res.text) ips = sub_dom.xpath("http://tbody/tr/td[1]/text()") ports = sub_dom.xpath("http://tbody/tr/td[2]/text()") types = sub_dom.xpath("http://tbody/tr/td[4]/text()") sleep(3) sub_res.close() for ip,port,type in zip(ips, ports,types): proxies_http = {} proxies_https= {} http = "http://" + ip + ":" + port https = "https://" + ip + ":" + port #分別存儲http和https兩種 proxies_http["http"] = http http_list.append(proxies_http) proxies_https["https"] = https https_list.append(proxies_https) return http_list,https_listif __name__ == "__main__": http_list,https_list = getip() print(http_list) print(https_list)
二,完整代碼放在最后后面了,這里的 from getip import getip 就是前面獲取ip部分。
這里我收集數十個常用的請求頭,將它們與三十個IP隨機組合,共可以得到300個左右的組合。
這里我定義了三個函數用于實現(xiàn)功能。
biquge_get()函數:輸入搜索頁面的url,關于搜索的實現(xiàn)是修改url中的kw,在main函數中有體現(xiàn)。
--------------------------返回書籍首頁的url和書名。
get_list()函數:輸入biquge_get返回的url。
---------------------返回每個章節(jié)的url集合。
info_get()函數:輸入url,ip池,請求頭集,書名。
---------------------將每次的信息保存到本地。
info_get()函數中我定義四個變量a,b,c,d用于判斷每個章節(jié)是否有信息返回,在代碼中有寫足夠清晰的注釋。
這里我講一下我的思路,在for循環(huán)中,我循環(huán)的是章節(jié)長度的十倍。a,b,c的初始值都是0。
通過索引,url=li_list[a]可以請求每個章節(jié)內容,a的自增實現(xiàn)跳到下一個url。但是在大量的請求中也會有無法訪問的情況,所以在返回的信息 ’ text1 ‘ 為空的情況a-=1,那么在下一次循環(huán)是依舊會訪問上次沒有結果的url。
這里我遇到了一個坑,我在測試爬取的時候會打印a的值用于觀察,出現(xiàn)它一直打印同一個章節(jié)數‘340’直到循環(huán)結束的情況,此時我以為是無法訪問了。后來我找到網頁對照,發(fā)現(xiàn)這個章節(jié)本來就沒有內容,是空的,所以程序會一直卡在這里。所以我設置了另外兩個變量b,c。
1,使用變量b來存放未變化的a,若下次循環(huán)b與a相等,說明此次請求沒有成功,c++,因為某些頁面本身存在錯誤沒有數據,則需要跳過。
2,若c大于10,說明超過十次的請求,都因為一些緣由失敗了,則a++,跳過這一章節(jié),同時變量d減一,避免后續(xù)跳出循環(huán)時出現(xiàn)索引錯誤
最后是變量d,d的初始值設置為章節(jié)長度,d = len(li_list),a增加到與d相同時說明此時li_list的所有url都使用完了,那么就需要跳出循環(huán)。
然后就是將取出的數據保存了。
最后測試,一共1676章,初始速度大概一秒能下載兩章內容左右。
爬取完成,共計用了10分鐘左右。
import requestsfrom lxml import etreefrom getip import getipimport randomimport timeheaders= { "User-Agent":"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36" }"""kw輸入完成搜索,打印所有的搜索結果返回選擇的書籍的url"""def biquge_get(url): book_info = [] r = requests.get(url =url, headers = headers, timeout = 20 ) r.encoding = r.apparent_encoding html = etree.HTML(r.text) # 獲取搜索結果的書名 bookname = html.xpath("http://td[@class = "odd"]/a/text()") bookauthor = html.xpath("http://td[@class = "odd"]/text()") bookurl = html.xpath("http://td[@class = "odd"]/a/@href") print("搜索結果如下:/n") a = 1 b = 1 for i in bookname: print(str(a) + ":", i, "/t作者:", bookauthor[int(b - 1)]) book_info.append([str(a),i,bookurl[a-1]]) a = a + 1 b = b + 2 c = input("請選擇你要下載的小說(輸入對應書籍的編號):") book_name = str(bookname[int(c) - 1]) print(book_name, "開始檢索章節(jié)") url2 = html.xpath("http://td[@class = "odd"]/a/@href")[int(c) - 1] r.close() return url2,book_name"""輸入書籍的url,返回每一章節(jié)的url"""def get_list(url): r = requests.get(url = url, headers = headers, timeout = 20) r.encoding = r.apparent_encoding html = etree.HTML(r.text) # 解析章節(jié) li_list = html.xpath("http://*[@id="list"]/dl//a/@href")[9:] return li_list#請求頭集user_agent = [ "Mozilla/5.0 (compatible; Baiduspider/2.0; +http://www.baidu.com/search/spider.html)", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; AcooBrowser; .NET CLR 1.1.4322; .NET CLR 2.0.50727)", "Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 6.0; Acoo Browser; SLCC1; .NET CLR 2.0.50727; Media Center PC 5.0; .NET CLR 3.0.04506)", "Mozilla/4.0 (compatible; MSIE 7.0; AOL 9.5; AOLBuild 4337.35; Windows NT 5.1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)", "Mozilla/5.0 (Windows; U; MSIE 9.0; Windows NT 9.0; en-US)", "Mozilla/5.0 (compatible; MSIE 9.0; Windows NT 6.1; Win64; x64; Trident/5.0; .NET CLR 3.5.30729; .NET CLR 3.0.30729; .NET CLR 2.0.50727; Media Center PC 6.0)", "Mozilla/5.0 (compatible; MSIE 8.0; Windows NT 6.0; Trident/4.0; WOW64; Trident/4.0; SLCC2; .NET CLR 2.0.50727; .NET CLR 3.5.30729; .NET CLR 3.0.30729; .NET CLR 1.0.3705; .NET CLR 1.1.4322)", "Mozilla/4.0 (compatible; MSIE 7.0b; Windows NT 5.2; .NET CLR 1.1.4322; .NET CLR 2.0.50727; InfoPath.2; .NET CLR 3.0.04506.30)", "Mozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN) AppleWebKit/523.15 (KHTML, like Gecko, Safari/419.3) Arora/0.3 (Change: 287 c9dfb30)", "Mozilla/5.0 (X11; U; Linux; en-US) AppleWebKit/527+ (KHTML, like Gecko, Safari/419.3) Arora/0.6", "Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.2pre) Gecko/20070215 K-Ninja/2.1.1", "Mozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.9) Gecko/20080705 Firefox/3.0 Kapiko/3.0", "Mozilla/5.0 (X11; Linux i686; U;) Gecko/20070322 Kazehakase/0.4.5", "Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.9.0.8) Gecko Fedora/1.9.0.8-1.fc10 Kazehakase/0.5.6", "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/535.11 (KHTML, like Gecko) Chrome/17.0.963.56 Safari/535.11", "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_7_3) AppleWebKit/535.20 (KHTML, like Gecko) Chrome/19.0.1036.7 Safari/535.20", "Opera/9.80 (Macintosh; Intel Mac OS X 10.6.8; U; fr) Presto/2.9.168 Version/11.52"]"""參數:url,ip池,請求頭集,書名"""def info_get(li_list,ip_list,headers,book_name): print("共計"+str(len(li_list))+"章") """ a,用于計數,成功請求到html并完成后續(xù)的存寫數據才會繼續(xù)請求下一個url b,在循環(huán)中存放未經過信息返回存儲判斷的a,用于與下一次循環(huán)的a作比較,判斷a是否有變化 c,若超過10次b=a,c會自增,則說明應該跳過此章節(jié),同時d減一 d,章節(jié)長度 """ a = 0 b = 0 c = 0 d = len(li_list) fp = open("./"+str(book_name)+".txt", "w", encoding="utf-8") #這里循環(huán)了10倍次數的章節(jié),防止無法爬取完所有的信息。 for i in range(10*len(li_list)): url = li_list[a] #判斷使用http還是https if url[4:5] == "s": proxies = random.choice(ip_list[0]) else: proxies = random.choice(ip_list[1]) try: r = requests.get(url=url, headers={"User-Agent": random.choice(headers)}, proxies=proxies, timeout=5 ) r.encoding = r.apparent_encoding r_text = r.text html = etree.HTML(r_text) try: title = html.xpath("/html/body/div/div/div/div/h1/text()")[0] except: title = html.xpath("/html/body/div/div/div/div/h1/text()") text = html.xpath("http://*[@id="content"]/p/text()") text1 = [] for i in text: text1.append(i[2:]) """ 使用變量b來存放未變化的a,若下次循環(huán)b與a相等,說明此次請求沒有成功,c++,因為某些頁面本身存在錯誤沒有數據,則需要跳過。 若c大于10,說明超過十次的請求,都因為一些緣由失敗了,則a++,跳過這一章節(jié),同時變量d減一,避免后續(xù)跳出循環(huán)時出現(xiàn)索引錯誤 """ if b == a: c += 1 if c > 10: a += 1 c = 0 d -=1 b = a #a+1,跳到下一個url,若沒有取出信息則a-1.再次請求,若有數據返回則保存 a+=1 if len(text1) ==0: a-=1 else: fp.write("第"+str(a+1)+"章"+str(title) + ":/n" +"/t"+str(",".join(text1) + "/n/n")) print("《"+str(title)+"》","下載成功!") r.close() except EnvironmentError as e: pass # a是作為索引在li_list中取出對應的url,所以最后a的值等于li_list長度-1,并以此為判斷標準是否跳出循環(huán)。 if a == d: break fp.close()if __name__ == "__main__": kw = input("請輸入你要搜索的小說:") url = f"http://www.b520.cc/modules/article/search.php?searchkey={kw}" bookurl,book_name = biquge_get(url) li_list = get_list(bookurl) ip_list = getip() t1 = time.time() info_get(li_list,ip_list,user_agent,book_name) t2 = time.time() print("耗時"+str((t2-t1)/60)+"min")
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/119991.html
各位童鞋,大家好,我是? 在《?UI自動化工具輕松實現(xiàn)微信消息收發(fā)?朋友圈爬取??》文末給童鞋們布置了一個作業(yè),批量朋友圈爬取,不知道大伙們是否還有映像。 看到很多小伙伴們踴躍報名參與,不禁要給你們點個贊??紤]到很多小伙伴想做卻做出來,這…貼心的我這不就來給你們公布參考答案啦? 對相關基礎還不了解的童鞋請參考前文: 《?UI自動化工具輕松實現(xiàn)微信消息收發(fā)?朋友圈爬取??》 《???對比PyWi...
摘要:剛學爬蟲不久,迫不及待的找了一個網站練手,新筆趣閣一個小說網站。繼續(xù)學習,有改進方案的歡迎提出來,一起交流。 剛學Python爬蟲不久,迫不及待的找了一個網站練手,新筆趣閣:一個小說網站。 前提準備 安裝Python以及必要的模塊(requests,bs4),不了解requests和bs4的同學可以去官網看個大概之后再回來看教程 爬蟲思路 剛開始寫爬蟲的小白都有一個疑問,進行到什么時候...
摘要:所以要設計條隊列,保存商品信息。數據更新問題有新商品進來,直接插入即可,如果是舊商品,那要不要更新數據庫里的內容呢一般來說是可以更新的,但有種情況例外,就是你的數據庫會有人去編輯的情況。 場景:爬取某商城的部分商品。 隊列設計 這里至少需要爬取2種資源,一種是商品列表,一種是商品信息。所以要設計1條隊列,保存商品信息URL。 爬蟲1定期爬前N個列表頁 URL,把里面的商品信息URL爬下...
摘要:把獲取到的下載視頻的存放在數組中也可寫入文件中,通過調用迅雷接口,進行自動下載。 把獲取到的下載視頻的url存放在數組中(也可寫入文件中),通過調用迅雷接口,進行自動下載。(請先下載迅雷,并在其設置中心的下載管理中設置為一鍵下載)實現(xiàn)代碼如下: from bs4 import Bea...
閱讀 869·2021-11-19 11:29
閱讀 3361·2021-09-26 10:15
閱讀 2871·2021-09-22 10:02
閱讀 2443·2021-09-02 15:15
閱讀 1981·2019-08-30 15:56
閱讀 2421·2019-08-30 15:54
閱讀 2926·2019-08-29 16:59
閱讀 644·2019-08-29 16:20