一個(gè)簡(jiǎn)易的Java多頁(yè)面隊(duì)列爬蟲(chóng)程序
之前寫(xiě)過(guò)很多單頁(yè)面python爬蟲(chóng),感覺(jué)python還是很好用的,這里用java總結(jié)一個(gè)多頁(yè)面的爬蟲(chóng),迭代爬取種子頁(yè)面的所有鏈接的頁(yè)面,全部保存在tmp路徑下?! ?/p>
一、 序言
實(shí)現(xiàn)這個(gè)爬蟲(chóng)需要兩個(gè)數(shù)據(jù)結(jié)構(gòu)支持,unvisited隊(duì)列(priorityqueue:可以適用pagerank等算法計(jì)算出url重要度)和visited表(hashset:可以快速查找url是否存在);隊(duì)列用于實(shí)現(xiàn)寬度優(yōu)先爬取,visited表用于記錄爬取過(guò)的url,不再重復(fù)爬取,避免了環(huán)。java爬蟲(chóng)需要的工具包有httpclient和htmlparser1.5,可以在maven repo中查看具體版本的下載。
1、目標(biāo)網(wǎng)站:新浪 http://www.sina.com.cn/
2、結(jié)果截圖:

下面說(shuō)說(shuō)爬蟲(chóng)的實(shí)現(xiàn),后期源碼會(huì)上傳到github中,需要的朋友可以留言:
二、爬蟲(chóng)編程
1、創(chuàng)建種子頁(yè)面的url
MyCrawler crawler = new MyCrawler();
crawler.crawling(new String[]{"http://www.sina.com.cn/"});
2、初始化unvisited表為上面的種子url
LinkQueue.addUnvisitedUrl(seeds[i]);
3、最主要的邏輯實(shí)現(xiàn)部分:在隊(duì)列中取出沒(méi)有visit過(guò)的url,進(jìn)行下載,然后加入visited的表,并解析改url頁(yè)面上的其它url,把未讀取的加入到unvisited隊(duì)列;迭代到隊(duì)列為空停止,所以這個(gè)url網(wǎng)絡(luò)還是很龐大的。注意,這里的頁(yè)面下載和頁(yè)面解析需要java的工具包實(shí)現(xiàn),下面具體說(shuō)明下工具包的使用。
while(!LinkQueue.unVisitedUrlsEmpty()&&LinkQueue.getVisitedUrlNum()<=1000)
{
//隊(duì)頭URL出隊(duì)列
String visitUrl=(String)LinkQueue.unVisitedUrlDeQueue();
if(visitUrl==null)
continue;
DownLoadFile downLoader=new DownLoadFile();
//下載網(wǎng)頁(yè)
downLoader.downloadFile(visitUrl);
//該 url 放入到已訪問(wèn)的 URL 中
LinkQueue.addVisitedUrl(visitUrl);
//提取出下載網(wǎng)頁(yè)中的 URL
Set<String> links=HtmlParserTool.extracLinks(visitUrl,filter);
//新的未訪問(wèn)的 URL 入隊(duì)
for(String link:links)
{
LinkQueue.addUnvisitedUrl(link);
}
}
4、下面html頁(yè)面的download工具包
public String downloadFile(String url) {
String filePath = null;
/* 1.生成 HttpClinet 對(duì)象并設(shè)置參數(shù) */
HttpClient httpClient = new HttpClient();
// 設(shè)置 Http 連接超時(shí) 5s
httpClient.getHttpConnectionManager().getParams().setConnectionTimeout(
5000);
/* 2.生成 GetMethod 對(duì)象并設(shè)置參數(shù) */
GetMethod getMethod = new GetMethod(url);
// 設(shè)置 get 請(qǐng)求超時(shí) 5s
getMethod.getParams().setParameter(HttpMethodParams.SO_TIMEOUT, 5000);
// 設(shè)置請(qǐng)求重試處理
getMethod.getParams().setParameter(HttpMethodParams.RETRY_HANDLER,
new DefaultHttpMethodRetryHandler());
/* 3.執(zhí)行 HTTP GET 請(qǐng)求 */
try {
int statusCode = httpClient.executeMethod(getMethod);
// 判斷訪問(wèn)的狀態(tài)碼
if (statusCode != HttpStatus.SC_OK) {
System.err.println("Method failed: "
+ getMethod.getStatusLine());
filePath = null;
}
/* 4.處理 HTTP 響應(yīng)內(nèi)容 */
byte[] responseBody = getMethod.getResponseBody();// 讀取為字節(jié)數(shù)組
// 根據(jù)網(wǎng)頁(yè) url 生成保存時(shí)的文件名
filePath = "temp\\"
+ getFileNameByUrl(url, getMethod.getResponseHeader(
"Content-Type").getValue());
saveToLocal(responseBody, filePath);
} catch (HttpException e) {
// 發(fā)生致命的異常,可能是協(xié)議不對(duì)或者返回的內(nèi)容有問(wèn)題
System.out.println("Please check your provided http address!");
e.printStackTrace();
} catch (IOException e) {
// 發(fā)生網(wǎng)絡(luò)異常
e.printStackTrace();
} finally {
// 釋放連接
getMethod.releaseConnection();
}
return filePath;
}
5、html頁(yè)面的解析工具包:
public static Set<String> extracLinks(String url, LinkFilter filter) {
Set<String> links = new HashSet<String>();
try {
Parser parser = new Parser(url);
parser.setEncoding("gb2312");
// 過(guò)濾 <frame >標(biāo)簽的 filter,用來(lái)提取 frame 標(biāo)簽里的 src 屬性所表示的鏈接
NodeFilter frameFilter = new NodeFilter() {
public boolean accept(Node node) {
if (node.getText().startsWith("frame src=")) {
return true;
} else {
return false;
}
}
};
// OrFilter 來(lái)設(shè)置過(guò)濾 <a> 標(biāo)簽,和 <frame> 標(biāo)簽
OrFilter linkFilter = new OrFilter(new NodeClassFilter(
LinkTag.class), frameFilter);
// 得到所有經(jīng)過(guò)過(guò)濾的標(biāo)簽
NodeList list = parser.extractAllNodesThatMatch(linkFilter);
for (int i = 0; i < list.size(); i++) {
Node tag = list.elementAt(i);
if (tag instanceof LinkTag)// <a> 標(biāo)簽
{
LinkTag link = (LinkTag) tag;
String linkUrl = link.getLink();// url
if (filter.accept(linkUrl))
links.add(linkUrl);
} else// <frame> 標(biāo)簽
{
// 提取 frame 里 src 屬性的鏈接如 <frame src="test.html"/>
String frame = tag.getText();
int start = frame.indexOf("src=");
frame = frame.substring(start);
int end = frame.indexOf(" ");
if (end == -1)
end = frame.indexOf(">");
String frameUrl = frame.substring(5, end - 1);
if (filter.accept(frameUrl))
links.add(frameUrl);
}
}
} catch (ParserException e) {
e.printStackTrace();
}
return links;
}
6、未訪問(wèn)頁(yè)面使用PriorityQueue帶偏好的隊(duì)列保存,主要是為了適用于pagerank等算法,有的url忠誠(chéng)度更高一些;visited表采用hashset實(shí)現(xiàn),注意可以快速查找是否存在;
public class LinkQueue {
//已訪問(wèn)的 url 集合
private static Set visitedUrl = new HashSet();
//待訪問(wèn)的 url 集合
private static Queue unVisitedUrl = new PriorityQueue();
//獲得URL隊(duì)列
public static Queue getUnVisitedUrl() {
return unVisitedUrl;
}
//添加到訪問(wèn)過(guò)的URL隊(duì)列中
public static void addVisitedUrl(String url) {
visitedUrl.add(url);
}
//移除訪問(wèn)過(guò)的URL
public static void removeVisitedUrl(String url) {
visitedUrl.remove(url);
}
//未訪問(wèn)的URL出隊(duì)列
public static Object unVisitedUrlDeQueue() {
return unVisitedUrl.poll();
}
// 保證每個(gè) url 只被訪問(wèn)一次
public static void addUnvisitedUrl(String url) {
if (url != null && !url.trim().equals("")
&& !visitedUrl.contains(url)
&& !unVisitedUrl.contains(url))
unVisitedUrl.add(url);
}
//獲得已經(jīng)訪問(wèn)的URL數(shù)目
public static int getVisitedUrlNum() {
return visitedUrl.size();
}
//判斷未訪問(wèn)的URL隊(duì)列中是否為空
public static boolean unVisitedUrlsEmpty() {
return unVisitedUrl.isEmpty();
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- java利用delayedQueue實(shí)現(xiàn)本地的延遲隊(duì)列
- Java 隊(duì)列實(shí)現(xiàn)原理及簡(jiǎn)單實(shí)現(xiàn)代碼
- 剖析Java中阻塞隊(duì)列的實(shí)現(xiàn)原理及應(yīng)用場(chǎng)景
- 詳解Java消息隊(duì)列-Spring整合ActiveMq
- 解析Java中的隊(duì)列和用LinkedList集合模擬隊(duì)列的方法
- java結(jié)合WebSphere MQ實(shí)現(xiàn)接收隊(duì)列文件功能
- java使用數(shù)組和鏈表實(shí)現(xiàn)隊(duì)列示例
- Java 阻塞隊(duì)列詳解及簡(jiǎn)單使用
- Java消息隊(duì)列的簡(jiǎn)單實(shí)現(xiàn)代碼
- Java并發(fā)編程之阻塞隊(duì)列詳解
- java多線程消息隊(duì)列的實(shí)現(xiàn)代碼
- Java延遲隊(duì)列原理與用法實(shí)例詳解
相關(guān)文章
Java 線程的優(yōu)先級(jí)(setPriority)案例詳解
這篇文章主要介紹了Java 線程的優(yōu)先級(jí)(setPriority)案例詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08
java集合中的迭代器Iterator和數(shù)組內(nèi)置方法及常見(jiàn)的報(bào)錯(cuò)解決方案
文章介紹了Java集合框架中迭代器(Iterator)的使用,以及數(shù)組和集合的內(nèi)置方法,重點(diǎn)解釋了在遍歷集合時(shí)刪除元素時(shí)可能出現(xiàn)的`ConcurrentModificationException`異常,并說(shuō)明了如何正確地使用迭代器來(lái)刪除集合中的元素,感興趣的朋友跟隨小編一起看看吧2025-02-02
springboot Controller直接返回String類(lèi)型帶來(lái)的亂碼問(wèn)題及解決
文章介紹了在Spring Boot中,當(dāng)Controller直接返回String類(lèi)型時(shí)可能出現(xiàn)的亂碼問(wèn)題,并提供了解決辦法,通過(guò)在`application.yaml`中設(shè)置請(qǐng)求和響應(yīng)的編碼格式,并在自定義配置類(lèi)中進(jìn)行配置,可以有效解決這一問(wèn)題2024-11-11
使用Spring Boot創(chuàng)建Web應(yīng)用程序的示例代碼
本篇文章主要介紹了使用Spring Boot創(chuàng)建Web應(yīng)用程序的示例代碼,我們將使用Spring Boot構(gòu)建一個(gè)簡(jiǎn)單的Web應(yīng)用程序,并為其添加一些有用的服務(wù),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05
Spring aop+反射實(shí)現(xiàn)電話號(hào)加密
線上項(xiàng)目涉及大量查詢接口中,存在電話號(hào)明文展示不合規(guī)的問(wèn)題。如果對(duì)每個(gè)接口返回結(jié)果中電話號(hào)相關(guān)字段修改相關(guān)代碼邏輯,則工作量較大花費(fèi)時(shí)間多。因此設(shè)計(jì)電話號(hào)加密注解,減少工作量。2021-06-06
IDEA運(yùn)行SpringBoot項(xiàng)目的圖文教程
本文主要介紹了IDEA運(yùn)行SpringBoot項(xiàng)目的圖文教程,文中通過(guò)圖文介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-05-05
Java多線程Thread類(lèi)的使用及注意事項(xiàng)
這篇文章主要介紹了Java多線程Thread類(lèi)的使用及注意事項(xiàng),在java標(biāo)準(zhǔn)庫(kù)中提供了一個(gè)Thread類(lèi)來(lái)表示/操作線程,Thread類(lèi)也可以視為是java標(biāo)準(zhǔn)庫(kù)提供的API2022-06-06
java過(guò)濾html標(biāo)簽獲取純文本信息的實(shí)例
今天小編就為大家分享一篇java過(guò)濾html標(biāo)簽獲取純文本信息的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-05-05
java實(shí)現(xiàn)table添加右鍵點(diǎn)擊事件監(jiān)聽(tīng)操作示例
這篇文章主要介紹了java實(shí)現(xiàn)table添加右鍵點(diǎn)擊事件監(jiān)聽(tīng)操作,結(jié)合實(shí)例形式分析了Java添加及使用事件監(jiān)聽(tīng)相關(guān)操作技巧,需要的朋友可以參考下2018-07-07

