成千上萬個站點,日資料過億的大規模爬蟲是怎麼實現的?
我們身邊接觸最頻繁、同時也是最大的爬蟲莫過於幾大搜尋引擎。但是搜尋引擎的爬取方式和我們爬蟲工程師接觸的方式差異比較大,沒有太大的參考價值,我們今天要講的是輿情方向的爬蟲(架構以及關鍵技術原理),主要涉及:
1.網頁文字智慧提取;
2.分散式爬蟲;
3.爬蟲 DATA/URL 去重;
4.爬蟲部署;
5.分散式爬蟲排程;
6.自動化渲染技術;
7.訊息佇列在爬蟲領域的應用;
8.各種各樣形式的反爬蟲;
請大家買好瓜子、搬好凳子坐下學習,並準備好爭奪文末贈送的獎品!
一、網頁文字智慧提取
輿情其實就是輿論情況,要掌握輿情,那麼就必須掌握足夠多的內容資訊。除了一些開放了商業介面的大型內容/社交類平臺(例如微博)之外,其他都需要依靠爬蟲去採集。因此,輿情方向的爬蟲工程師需要面對的是千千萬萬個內容和結構都不同的站點。我們用一個圖來表示他們面對的問題:
沒錯,他們的採集器必須要能夠適配千千萬萬個站點的結構,從風格迥異的 HTML 文字中提取出主體內容——標題、正文、釋出時間和作者。
如果是你,你會用什麼樣的設計來滿足業務需求呢?
曾經我也設想過這樣的問題,在技術群裡也看到有群友提出類似的問題,但是很難得到滿意的答案。有的人說:
1.用歸類法,相似的內容歸類到一起,然後給一類內容配置提取規則;
2.用正則,提取指定標籤中的內容;
3.用深度學習,NLP 語義分析出哪裡是有意義的內容,提取出來;
4.用計算機視覺,讓人去點選,然後按照頁面相似度分類提取(其實就是歸類法的自動化版本);
5.用演算法,計算出文字的密度,然後提取;
總之各種各樣的想法層出不窮,但是最後都沒有聽到實際應用的訊息。目前來說,大部分公司使用的都是人工配置 XPATH 規則的方式,採集的時候通過網址來匹配對應的提取規則,然後呼叫規則來實現多站點的爬取。這種方法很有效,而且在企業中應用已久,比較穩定,但缺點也顯而易見——費時間、費人工、費錢!
偶有一天,我在微信技術群裡看到有人(優秀的 Python 工程師青南)發表了一個用於自動化提取文字的演算法庫,GeneralNewsExtractor[1] (以下簡稱 GNE)。這個庫參考了武漢郵電科學研究院洪鴻輝、丁世濤、黃傲、郭致遠等人編寫的論文——《基於文字及符號密度的網頁正文提取方法》,並在論文的基礎上用 Python 程式碼進行了具體實現,也就是 GNE。它的原理是通過提取網頁 DOM 中的文字以及其中的標點符號,以文字中標點符號的密度作為基礎,使用演算法從一句話延伸到一段文字和一篇文章。
GNE 能夠有效排除正文以外的的廣告、推薦欄、介紹欄等“噪聲”內容,準確識別出網頁正文,且識別率高達 99%(測試選用的內容為國內主流門戶/媒體平臺中的文章)。
GNE 的具體演算法細節以及原始碼解析請翻閱《Python3 網路爬蟲寶典》第 5 章。
有了它,基本上就可以解決 90% 以上的輿情方向爬蟲解析的需求,剩下的 10% 可以基於提取規則進行鍼對性調整或者完全定製化開發,解放了一大波 XPATH 工程師。
二、爬蟲 DATA/URL 去重
輿情業務必須緊盯網站是否有新的內容釋出,要求是越快越好,但由於各項軟硬體限制,通常會要求在 30 分鐘內或者 15 分鐘內監聽到新內容。要實現對目標網站內容變化的監聽,那麼我們可以選擇的比較好的方式就是輪詢。不停地訪問網頁,並且判斷是否有“新內容”出現,如果有的話就執行爬取,沒有“新內容”就不爬取。
那麼問題來了,應用程式如何知道哪些內容是“新的”、哪些內容又是“舊的”的呢?
問題拆解一下,“新內容”就是沒有爬過的內容。這時候我們需要用一個東西來記錄這篇文章是否被爬取過,每次有要爬取的文章時就比對一下,這就是解決這個問題的辦法。
那又依靠什麼來進行比對呢?
我們都知道文章的 URL 幾乎都是不變且不會重複的,因此可以選擇文章的 URL 作為判定的依據,也就是把爬取過的 URL 放到一個類似列表一樣的容器裡儲存起來,每次有待爬取的 URL 就判斷它是否已經被存在容器裡,如果在就說明已經爬過了,直接丟棄,進入下一個 URL 的判斷流程。整體邏輯就像下面這張圖一樣:
這就是爬蟲領域的“去重”。實際上去重可以粗略的分為內容(DATA)去重和連結(URL)去重,這裡我們講的只是輿情方向的去重需求,如果是電商方向的去重,那麼就不能用 URL 作為判斷依據,因為電商爬蟲(例如比價軟體)的目的主要是判斷價格的變化,這時候判斷變化的依據應該是商品的關鍵資訊(例如價格、折扣),也就是 DATA 去重。
去重的原理明白了,那用什麼東西作為存放去重依據的容器呢?MySQL?Redis?MongoDB?記憶體?實際上大部分工程師都選擇 Redis 作為存放去重依據的容器,但實際上 MySQL、MongoDB 和記憶體都是可以充當容器的,至於為什麼會選擇 Redis,它又比其他資料儲存好在哪裡?你可以翻閱《Python3 網路爬蟲寶典》的第 3 章。
三、分散式爬蟲
無論是輿情方向的爬蟲還是電商方向的爬蟲,要承擔的爬取量都是非常大的。少則每日百萬資料,多則每日數十億資料。以往大家瞭解的單機爬蟲,在效能和資源方面都無法滿足需求。既然 1 個滿足不了,那就 10 個、100 個!這就是分散式爬蟲出現的背景。
眾所周知,分散式和單機要面對的問題是有差異的,除了業務目標是相同的之外,分散式還要考慮多個個體之間的協作,尤其是資源的共享和競爭。
在只有 1 個爬蟲應用的時候,也只有它 1 個讀取待爬佇列、只有 1 個儲存資料、只有它 1 個判斷 URL 是否重複。但有幾十幾百個爬蟲應用的時候,就需要區分先後順序,避免出現多個爬蟲應用訪問同一個 URL 的情況(因為這不僅浪費時間,還浪費資源)。而且,只有 1 個爬蟲應用的時候只需要把它放在 1 臺計算機(伺服器)上執行就可以了,但是爬蟲應用突然變得這麼多,又應該如何部署到不同的計算機上呢?手動一個個上傳,然後一個個啟動嗎?
資源問題
先說資源共享和競爭的情況,為了解決 URL 待爬佇列和已經爬佇列的共享,那麼必須將佇列(也就是上面提到的存放 URL 的容器)放到一個可以公開(多個爬蟲應用)訪問的地方,例如部署在伺服器上的 Redis。
這時候又出現一個新狀況,隨著資料量越來越大,要儲存的 URL 越來越多,後面很有可能出現因為儲存空間需求過大而導致成本遞增的問題。因為 Redis 是利用記憶體來儲存資料的,所以存放的 URL 越多就需要越多的記憶體,而記憶體又是硬體裝置裡價格相對較高的硬體,所以不得不考慮這個問題。
好在一個叫做布隆的人發明了一種演算法——Bloom Filter(布隆過濾器),這種演算法通過雜湊對映的方式來標記一個物件(這裡是 URL)是否存在,這樣可以將記憶體的佔用率大大降低,按 1 億條長度為 32 字元的 URL MD5 值來計算,使用 Bloom Filter 前後的差距大約在 30倍。關於 Bloom Filter 的演算法原理和程式碼實現解讀可翻閱《Python3 網路爬蟲寶典》第 3 章 。
部署問題
一個個檔案上傳,一次次手動執行爬蟲實在是太累了。你可以向運維同事尋求技術支援,但也可以自己探尋這些能夠減輕你工作量的自動化部署方式。目前業內知名的持續整合和部署莫過於 GitLab 的 GitLab Runner 和 GitHub Action,又或者是藉助 K8S 的容器化來實現。但它們只能幫助你實現部署和啟動,而爬蟲應用的一些管理功能就指望不上了。遂,今天要給大家介紹的是另一種實現方式——使用 Crawlab。
Crawlab 是一款由知名外企工程師開發的分散式爬蟲管理平臺,它不僅支援 Python 語言編寫的爬蟲,幾乎可以相容大部分程式語言和應用程式。藉助 Crawlab,我們可以將爬蟲應用分散到不同的計算機(伺服器)上,而且能夠在視覺化介面設定定時任務、檢視平臺上爬蟲應用的狀態以及環境依賴等資訊。具體如下圖所示:
面對一款如此實用的平臺工具,作為工程師的我們不禁想問:
1.它是如何把檔案分散到不同計算機的?
2.它如何實現不同計算機(多節點)之間通訊的?
3.它是如何實現多語言相容的?
4.……
其中我們比較關注的多節點通訊是藉助 Redis 實現的,檔案分散同步是藉助 MongoDB 實現的。更多細節可翻閱《Python3 網路爬蟲寶典》 第 6 章。
除了這樣的平臺之外,Python 爬蟲工程師常常接觸的莫過於 Scrapy 框架以及相關衍生的庫。Scrapy 團隊官方開發了一個名為 Scrapyd 的庫,它專門用來部署 Scrapy 框架開發的爬蟲應用。在部署 Scrapy 應用時,我們通常只需要執行 1 行命令就可以把爬蟲程式部署到伺服器上。你想不想知道背後的邏輯:
1.程式以什麼樣的形式上傳到伺服器的?
2.程式在伺服器上如何執行的?
3.為什麼可以檢視到每個任務執行的開始時間和結束時間?
4.中途取消任務執行的功能是怎麼實現的?
5.它的版本控制是怎麼實現的?
6.如果不是 Scrapy 框架編寫的 Python 應用,能實現像上面幾點那樣的監控和操作嗎?
實際上 Scrapy 應用會被打包成為一個字尾為“.egg” 的壓縮包,以 HTTP 的形式上傳到伺服器上。當服務端程式需要執行這個程式時,先將它複製到作業系統的臨時資料夾,執行時將其匯入到當前 Python 環境,執行完畢後刪除該檔案。至於它的執行時間和中斷操作,實際上藉助了 Python 程式介面,具體細節翻閱《Python3 網路爬蟲寶典》 第 6 章。
四、自動化渲染技術
為了實現炫酷的效果,或者說為了節省靜態資源對頻寬的佔用,很多網站都是藉助 JavaScript 來實現對頁面內容的優化。Python 程式本身是無法解釋 JavaScript 和 HTML 程式碼的,因此無法獲得我們在瀏覽器中“看到”,但實際上並不是“真實存在”的內容。
因為這些內容都是由瀏覽器渲染出來的,只存在於瀏覽器中,HTML 文件裡面還是那些文字、JavaScript 檔案中還是那些程式碼,圖片、視訊和那些特效並不會出現在程式碼中,我們看到的一切都是瀏覽器的功勞。
由於 Python 也無法獲取瀏覽器渲染後的內容,所以當我們像往常一樣寫程式碼爬取上面的資料時,就會發現拿到的資料和看到的並不一樣,任務它就失敗了。
這時候我們就需要用到自動化渲染技術了,實際上像 Chrome 和 FireFox 這樣的瀏覽器都開放了介面,允許其他程式語言按照協議規範操控瀏覽器。基於這樣的技術背景,有團隊開發出了像 Selenium 和 Puppeteer 這樣的工具,然後我們就可以用 Python (其他語言也可以)程式碼來操作瀏覽器了。讓瀏覽器幫助我們做一些使用者名稱密碼輸入、登入按鈕點選、文字和圖片渲染、驗證碼滑動等操作,從而打破 Python 與瀏覽器本身的差異壁壘,藉助瀏覽器渲染內容後再返回給 Python 程式,然後拿到和我們在網頁上看到的一樣的內容。
除了瀏覽器,APP 也有類似的情況。具體操作實踐和案例細節可翻閱《Python3 網路爬蟲寶典》 第 2 章。
五、訊息佇列在爬蟲領域的應用
之前的描述中,我們並沒有提到爬取時候的細節。假設這樣一個正常的爬蟲場景:爬蟲先訪問網站的文章列表頁,然後根據列表頁的 URL 進入詳情頁進行爬取。這裡要注意,文章詳情頁的數量一定是比列表頁的數量多 N 倍的,如果列表展示的是 20 條內容,那麼就是多 20 倍。
如果我們需要爬取的網站很多,那麼就會用到分散式爬蟲。如果分散式爬蟲只是把 1 個爬蟲程式複製出 N 份來執行,那麼就會出現資源分配不均衡的情況,因為在上面提到的這種情況下,每 1 個爬蟲都需要這麼幹活。實際上我們可以有更好的搭配方式,讓它們的資源得到最大利用。例從列表頁到詳情頁可以抽象為生產者和消費者模型:
4 號和 5 號爬蟲應用只負責將列表頁中抽取詳情頁的 URL,然後推送到一個佇列中,另外幾個爬蟲程式從佇列中取出詳情頁的 URL 進行爬取。當列表頁和詳情頁數量差距比較大的時候,我們可以增加右側的爬蟲程式數量,差距較小的時候就減少右側的爬蟲程式(或者增加左側的爬蟲程式,具體視情況定)。
左側的爬蟲程式相對於佇列這條“資料採集生產線”來說,它就是生產者,右側爬蟲程式的就是消費者。有了這樣的結構,我們就可以根據實際情況對生產者或者消費者的熟練進行調整,實現資源的最大化利用。
另外一個好處是當生產者拿到的 URL 越來越多,但消費者一時消費不過來時,URL 會一直存放在佇列中,等消費能力上升時就能夠再次實現均衡。有了這樣的生產線,我們就不用擔心一下突然湧來很多的 URL 或者一下突然把佇列的 URL 消費一空,佇列這種削峰填谷的能力除了在後端應用中大放異彩之外,在爬蟲應用中也發揮了很大的作用。
關於爬蟲(以及分散式爬蟲)程式接入訊息佇列的具體實現和細節可翻閱《Python3 網路爬蟲寶典》 第 4 章。
六、各種各樣形式的反爬蟲
你想要我偏不給!
網站可不會輕易讓你爬取站點上面的內容,它們往往會從網路協議、瀏覽器特徵、程式語言差異、人機差異等方面給爬蟲工程師設定障礙,常見的有滑塊驗證碼、拼圖驗證碼、封 IP、檢查 COOKIE、要求登入、設定複雜的加密邏輯、混淆前端程式碼等。
水來土掩、兵來將擋!爬蟲工程師與目標網站的工程師你來我往的過招就像兵家爾虞我詐一般精彩。《Python3 反爬蟲原理與繞過實戰》一書囊括了市面上 80% 以上的反爬蟲手段和爬蟲技巧,詳細解讀雙方所用招術,使各位看客從中學到不少使用招式。具體細節可翻閱該書,領略技術領域的江湖!
小結
今天我們一起學習了日資料量過億的大規模爬蟲實踐路上的關鍵技術點,包括文字智慧提取、分散式爬蟲、爬蟲部署和排程、去重、自動化渲染等。學會了這些技術並融會貫通,那麼實現日資料過億的爬蟲就不是問題了。
這些經驗都來自一線爬蟲工程師,同時這些技術和設計都經過了長期的工作驗證,能夠直接應用在工作當中。
活動
上面多次提到了《Python3 網路爬蟲寶典》,小編購買了5本書用來感謝大家對小編的支援。領取方式,公眾號後臺輸入:爬蟲
送書5本,領取方式如下:
本公眾號,後臺輸入關鍵字:爬蟲
相關文章
- 大規模非同步新聞爬蟲的實現思路非同步爬蟲
- 大規模非同步新聞爬蟲:實現一個同步定向新聞爬蟲非同步爬蟲
- 大規模非同步新聞爬蟲的分散式實現非同步爬蟲分散式
- 大規模非同步新聞爬蟲: 用asyncio實現非同步爬蟲非同步爬蟲
- 每秒採集幾十萬資料的大規模分散式爬蟲是如何煉成的?分散式爬蟲
- 不踩坑的Python爬蟲:如何在一個月內學會爬取大規模資料Python爬蟲
- 每秒幾十萬的大規模網路爬蟲是如何煉成的?爬蟲
- 大規模非同步新聞爬蟲:實現一個更好的網路請求函式非同步爬蟲函式
- Python爬蟲是如何實現的?Python爬蟲
- 什麼是爬蟲?Python爬蟲的工作流程怎樣?爬蟲Python
- 大規模爬蟲為什麼要管理DNS快取爬蟲DNS快取
- 大規模非同步新聞爬蟲:簡單的百度新聞爬蟲非同步爬蟲
- 什麼是爬蟲?爬蟲的工作原理是什麼呢爬蟲
- 面試—html語義化,SEO的原理,什麼是爬蟲、怎麼去寫一個爬蟲面試HTML爬蟲
- 大規模非同步新聞爬蟲:實現功能強大、簡潔易用的網址池(URL Pool)非同步爬蟲
- 大規模非同步新聞爬蟲: 讓MySQL 資料庫操作更方便非同步爬蟲MySql資料庫
- 想做個防爬蟲的功能怎麼做爬蟲
- 大規模非同步新聞爬蟲:網頁正文的提取非同步爬蟲網頁
- JVM的ServerSocket是怎麼實現的(上)JVMServer
- 爬蟲資料是如何收集和整理的?爬蟲
- 大資料爬蟲專案實戰教程大資料爬蟲
- IPIDEA大盤點,藉助網路爬蟲抓取資料的作用?Idea爬蟲
- 大資料是怎麼知道你去過新發地的?大資料
- 爬蟲是如何被網站識別的?爬蟲網站
- 網路爬蟲——Urllib模組實戰專案(含程式碼)爬取你的第一個網站爬蟲網站
- 在大規模 Kubernetes 叢集上實現高 SLO 的方法
- 大資料的處理是怎樣的過程大資料
- 爬蟲在大資料時代的應用爬蟲大資料
- Serverless 在大規模資料處理的實踐Server
- 多執行緒爬蟲實現(上)執行緒爬蟲
- python爬蟲簡歷專案怎麼寫_爬蟲專案咋寫,爬取什麼樣的資料可以作為專案寫在簡歷上?...Python爬蟲
- ReactPHP 爬蟲實戰:下載整個網站的圖片ReactPHP爬蟲網站
- 個人分享 | 我的常規爬蟲流程爬蟲
- 防不勝防 成千上萬的 WordPress 網站被掛上惡意程式碼網站
- 什麼是大資料?大資料的產生、特點、用途大資料
- java實現一個簡單的爬蟲小程式Java爬蟲
- 大規模爬蟲系統面臨的主要挑戰及解決思路爬蟲
- 大資料叢集遷移的那一夜是怎麼過的大資料