说到爬虫,使用Java本身自带的URLConnection可以实现一些基本的抓取页面的功能,但是对于一些比较高级的功能,比如重定向的处理,HTML标记的去除,仅仅使用URLConnection还是不够的。

在这里我们可以使用HttpClient这个第三方jar包。

接下来我们使用HttpClient简单的写一个爬去百度的Demo:

package internet_worm.Demo1;

import java.io.FileOutputStream;  
import java.io.InputStream;  
import java.io.OutputStream;  

import org.apache.commons.httpclient.HttpClient;

import org.apache.commons.httpclient.HttpStatus;
import org.apache.commons.httpclient.methods.GetMethod; 

public class Demo2 {
    private static HttpClient httpClient =new HttpClient();
    /**
     *@param path
     *          目标网页的连接
     *@return   返回布尔值,表示是否正常下载目标页面
     *@throws Exception
     *          读取网页流或写入本地文件的IO异常 
     */
    public static boolean downloadPage(String path)throws  Exception{
        //定义输入输出流
        InputStream input =null;
        OutputStream output =null;
        String filename1 = path.substring(path.lastIndexOf('/')+1)+".html"; 
        //得到post方法
        GetMethod getMethod =new GetMethod(path);
        //执行,返回状态码
        int statusCode=httpClient.executeMethod(getMethod);
        //针对状态码进行处理
        //简单起见,只处理返回值为200的状态码
        if (statusCode == HttpStatus.SC_OK){
            input=getMethod.getResponseBodyAsStream(); 
            //通过对URL的得到的文件名
            String filename = path.substring(path.lastIndexOf('/')+1)+".html"; 
            //获得文件输出流
            output = new FileOutputStream(filename);
            //得到文件
            int tempByte=-1;
            while ((tempByte = input.read())>0){
                output.write(tempByte);
            }
            //关闭输入流
            if(input!=null){
                input.close();
            }
            //关闭输出流
            if(output!=null){
                output.close();
            }
            return true;
        }   
        return false;
    }

    public static void main(String[] args) {
        try {
            //抓取百度首页,输出
            Demo2.downloadPage("http://www.baidu.com");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

但是这样基本的爬虫是不能满足各色各样的爬虫需求的。

先来介绍宽度优先爬虫。

宽度优先相信大家都不陌生,简单说来可以这样理解宽度优先爬虫。

我们把互联网看作一张超级大的有向图,每一个网页上的链接都是一个有向边,每一个文件或没有链接的纯页面则是图中的终点:

java爬虫生成word文档 java写爬虫程序_html

宽度优先爬虫就是这样一个爬虫,爬走在这个有向图上,从根节点开始一层一层往外爬取新的节点的数据。

宽度遍历算法如下所示:

(1) 顶点 V 入队列。
(2) 当队列非空时继续执行,否则算法为空。
(3) 出队列,获得队头节点 V,访问顶点 V 并标记 V 已经被访问。
(4) 查找顶点 V 的第一个邻接顶点 col。
(5) 若 V 的邻接顶点 col 未被访问过,则 col 进队列。
(6) 继续查找 V 的其他邻接顶点 col,转到步骤(5),若 V 的所有邻接顶点都已经被访问过,则转到步骤(2)。

按照宽度遍历算法,上图的遍历顺序为:A->B->C->D->E->F->H->G->I,这样一层一层的遍历下去。

而宽度优先爬虫其实爬取的是一系列的种子节点,和图的遍历基本相同。

我们可以把需要爬取页面的URL都放在一个TODO表中,将已经访问的页面放在一个Visited表中:

java爬虫生成word文档 java写爬虫程序_apache_02

则宽度优先爬虫的基本流程如下:

(1) 把解析出的链接和 Visited 表中的链接进行比较,若 Visited 表中不存在此链接, 表示其未被访问过。
(2) 把链接放入 TODO 表中。
(3) 处理完毕后,从 TODO 表中取得一条链接,直接放入 Visited 表中。
(4) 针对这个链接所表示的网页,继续上述过程。如此循环往复。

下面我们就来一步一步制作一个宽度优先的爬虫。

首先,对于先设计一个数据结构用来存储TODO表, 考虑到需要先进先出所以采用队列,自定义一个Quere类:

package internet_worm.Demo2;

import java.util.LinkedList;

/**
 * 自定义队列累 保存TODO表
 * @author sky
 *
 */
public class Queue {
/**
 * 定义一个队列,使用LinkedList
 */
private LinkedList<Object> queue= new LinkedList<Object>();//入队列
/*
 * 将T加入到队列中
 */

public void enQueue(Object  t){
    queue.addLast(t);
}
/*
 * 移除队列中的第一项并将其返回
 */
public Object deQueue(){
    return queue.removeFirst();
}

/*
 * 返回队列是否为空
 * 
 */
public boolean isQueueEmpty(){
    return queue.isEmpty();
}
/**
 * 判断并返回队列是否包含T
 */
public boolean contians(Object t){
    return queue.contains(t);
}
/**
 * 判断并返回队列是否为空
 */
public boolean empty(){
    return queue.isEmpty();
}
}

还需要一个数据结构来记录已经访问过的 URL,即Visited表。

考虑到这个表的作用,每当要访问一个 URL 的时候,首先在这个数据结构中进行查找,如果当前的 URL 已经存在,则丢弃这个URL任务。

这个数据结构需要不重复并且能快速查找,所以选择HashSet来存储。

综上,我们另建一个SpiderQueue类来保存Visited表和TODO表:

package internet_worm.Demo2;

import java.util.HashSet;
import java.util.Set;

/*
 * 自定义类,保存Visited表和unVisited表
 */
public class SpiderQueue {
/*
 * 已访问的url集合,即Visited表
 */
private static Set<Object> visitedUrl = new HashSet<Object>();
/**
 * 添加到访问过的URL队列中
 */
public static void addVisitedUrl(String url){
    visitedUrl.add(url);
}
/*
 * 移除访问过的URL
 */
public static void removeVisitedUrl(String url){
    visitedUrl.remove(url);
}
/*
 * 获得已经访问的URL数目
 */
public static int getVisitedUrlNum(){
    return visitedUrl.size();
}
/*
 * 待访问的url集合,即unVisited表
 */
private static Queue unVisitedUrl = new Queue();
/*
 * 获得UnVisited队列
 */
public static Queue getUnVisitedUrl(){
    return unVisitedUrl;
}
/*
 * 未访问的unVisitedUrl出队列
 */
public static Object unVisitedUrlDeQueue(){
    return unVisitedUrl.deQueue();
}
/*
 * 保证添加url到unVisitedUrl的时候每个 URL只被访问一次
 */
public static void addUnvisitedUrl(String url){
    if(url != null && !url.trim().equals("") && !visitedUrl.contains(url)
            && !unVisitedUrl.contians(url))
        unVisitedUrl.enQueue(url);
}
/*
 * 判断未访问的URL队列中是否为空
 */
public static boolean unVisitedUrlsEmpty(){
    return unVisitedUrl.empty();
}

}

上面是一些自定义类的封装,接下来就是一个定义一个用来下载网页的工具类,我们将其定义为DownTool类:

package internet_worm.Demo2;

import java.io.DataOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;

import org.apache.commons.httpclient.DefaultHttpMethodRetryHandler;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpException;
import org.apache.commons.httpclient.HttpMethod;
import org.apache.commons.httpclient.HttpStatus;
import org.apache.commons.httpclient.methods.GetMethod;
import org.apache.commons.httpclient.methods.PostMethod;
import org.apache.commons.httpclient.params.HttpMethodParams;
import org.apache.commons.httpclient.util.HttpURLConnection;

public class Downtool {
    /**
     * 根据URL和网页类型生成需要保存的网页文件名了,去除URL中的非文件名字符
     */
    private String getFileNameByUrl(String url,String contentType){
        //移除"http://"这七个字符
        url = url.substring(7);
        //确认抓取到的页面text/html类型
        if(contentType.indexOf("html")!=-1){
            //把所有的url中的特殊符号转化成下划线
            url=url.replaceAll("[\\?/:*|<>\"]", "_")+".html";
        }else{
            url =url.replaceAll("[\\?/:*|<>\"]", "_")+"."+
            contentType.substring(contentType.lastIndexOf("/")+1);
        }
        return url;
    }
    /**
     * 保存网页字节到本地文件,filePath为要保存的文件的相对地址
     */
    private void saveToLocal(byte[] data,String filePath){
        try{
            File file=new File(filePath);
            DataOutputStream out = new DataOutputStream(new FileOutputStream(
                    new File(filePath)));
            for(int i=0;i<data.length;i++){
                out.write(data[i]);
            }
                out.flush();
                out.close();

        }catch(Exception e){
            e.printStackTrace();
        }
    }
    //下载URL指向的网页
    public String downlaodFile(String url){
        String filePath =null;
        //1.生成HttpClient对象设置参数
        HttpClient httpClient = new HttpClient();
        //设置HTTP连接超时5s
        httpClient.getHttpConnectionManager().getParams()
        .setConnectionTimeout(5000);
        //2.生成GetMethod对象并设置参数
        GetMethod getMethod = new GetMethod(url);
        //设置get请求5s
        getMethod.getParams().setParameter(HttpMethodParams.SO_TIMEOUT, 5000);
        //设置请求重试处理
        getMethod.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, new DefaultHttpMethodRetryHandler());
        //3.执行get请求
        try{
            int statusCode = httpClient.executeMethod(getMethod);
            //判断访问状态码
            if(statusCode!=HttpStatus.SC_OK){
                System.err.println("Method failed:"
                        + getMethod.getStatusLine());
                filePath =null;
            }
            //4.处理HTTP响应内容
            byte[] responseBody = getMethod.getResponseBody();//读取为字节数组
            //根据网页url生成保存时的文件名
            filePath ="temp\\"+getFileNameByUrl(url,
                    getMethod.getResponseHeader("Content-Type")
                    .getValue());
            saveToLocal(responseBody, filePath);
        }catch(HttpException e){
            //发生致命的异常,可能是协议不对或者返回的内容有问题
            System.out.println("请检查你的http地址是否正确");
            e.printStackTrace();
        } catch (IOException e) {
            //发生网络异常
            e.printStackTrace();
        }finally{
            //释放连接
            getMethod.releaseConnection();
        };
        return filePath;

    }
}

这里要写一个过滤器接口,保证从返回的HTML里提取的网页地址都是目标网站的下子网页,如我们现在要爬的是www.baidu.com,那我们就要定义是下载www.baidu.com开头的网页,而不是外链的网址。

package model;

public interface LinkFilter {
    public boolean accept(String url);
}

在这里我们需要一个HtmlParserTool类来处理Html标记(解析返回的HTML)
HtmlParserTool是一个很好的HTML解析库,具体用法可以查看

package internet_worm.Demo2;

import java.util.HashSet;
import java.util.Set;

import org.htmlparser.Node;
import org.htmlparser.NodeFilter;
import org.htmlparser.Parser;
import org.htmlparser.filters.NodeClassFilter;
import org.htmlparser.filters.OrFilter;
import org.htmlparser.tags.BodyTag;
import org.htmlparser.tags.Html;
import org.htmlparser.tags.LinkTag;
import org.htmlparser.util.NodeIterator;
import org.htmlparser.util.NodeList;
import org.htmlparser.util.ParserException;

import com.sun.xml.internal.xsom.impl.scd.ParseException;

import model.LinkFilter;

public class HtmlParserTool {
    //定义可能出现的HTML字符集
    private static final String oriEncode = "utf-8,gb2312,gbk,iso-8859-1";
    private static String encode=null;
    //获取一个网站上的链接,filter用来过滤链接
    public static Set<String> extracLinks(String url,LinkFilter filter){
        String encode=null;
        Set<String> links =new HashSet<String>();
        try{
            //处理字符组
            String[] encodes = oriEncode.split(",");
            for (int i = 0; i < encodes.length; i++) {
                Parser parser =new Parser(url);
                 parser.setEncoding(encodes[i]);
                 /*
                  * 字符集是否存在页面返回的内容里,判断出页面是形容哪种字符集解码
                  */

                 for (NodeIterator e = parser.elements(); e.hasMoreNodes();) {
                       Node node = (Node) e.nextNode();
                       if (node instanceof Html||node instanceof BodyTag) {
                           encode=encodes[i];
                           break;
                       }
                   }
                 if(encode !=null){
                     break;
                 }

               }

            Parser parser =new Parser(url);

            parser.setEncoding(encode);

            //过滤<frame>标签的filter,用来提取frame标签的src属性
            NodeFilter frameFilter = new NodeFilter(){
                private static final long serialVersionUID = 1L;
                //@Override
                public boolean accept(Node node) {
                    // TODO Auto-generated method stub
                    if(node.getText().startsWith("frame src=")){
                        return true;
                    }else{
                        return false;
                    }

                }
            };
        //OrFilter来设置过滤<a>标签和<frame>标签
        OrFilter linkFilter = new OrFilter(new NodeClassFilter(
                LinkTag.class),frameFilter);
        //得到所有经过过滤的标签
        NodeList list = parser.extractAllNodesThatMatch(linkFilter);
        for(int i = 0;i<list.size();i++){
            Node tag=list.elementAt(i);
            if(tag instanceof LinkTag)//<a>标签
            {
                LinkTag link=(LinkTag) tag;
                String linkUrl = link.getLink();//URL
                if(filter.accept(linkUrl))
                    links.add(linkUrl);
                }else//<frame>标签
                {
                //提取frame里src属性的链接,如<frame src="test.html"/>
                String frame =tag.getText();
                System.out.println(frame);
                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;

    }
}

最后我们来写个爬虫类调用前面的封装类和函数:

package internet_worm.Demo2;
import java.util.Set;

import org.omg.CORBA.PUBLIC_MEMBER;

import model.LinkFilter;
import internet_worm.Demo2.SpiderQueue;
public class BfsSpider {
    /**
     * 使用种子初始化URL队列
     */
    private void initCrawlerWithSeeds(String[] seeds){
        for(int i= 0; i <seeds.length;i++){
            SpiderQueue.addUnvisitedUrl(seeds[i]);
        }
    }
    //定义过滤器,提取以http://www.xxx.com开头的连接
    public void crawling(String[] seeds){
        LinkFilter filter = new LinkFilter(){
            public boolean accept(String url) {
            // TODO Auto-generated method stub
            //重写LinkFilter接口方法
                if(url.startsWith("http://www.baidu.com"))
                    return true;
                else
                    return false;
    }
    };
    //初始化URL队列
    initCrawlerWithSeeds(seeds);
    //循环条件:待抓取的链接不空切抓取的网页不多于100
    while (!SpiderQueue.unVisitedUrlsEmpty()
            && SpiderQueue.getVisitedUrlNum()<=100){
        //队头URL出队列
        String visitUrl =(String)SpiderQueue.unVisitedUrlDeQueue();
        if(visitUrl == null){
            continue;
        }
        Downtool downLoader= new Downtool();
        //下载网页
        downLoader.downlaodFile(visitUrl);
        //该URL放入已访问的URL中
        SpiderQueue.addVisitedUrl(visitUrl);
        //提取出下载网页中的URL入队
        Set<String> links =HtmlParserTool.extracLinks(visitUrl, filter);
        //新的未访问URL入队
        for(String link:links){
            SpiderQueue.addUnvisitedUrl(link);
        }
    }

}   
    //main方法入口
    public static void main(String[] args){
        BfsSpider crawler =new BfsSpider();
        crawler.crawling(new String[] {"http://www.baidu.com"});
    }
}

运行可以看到,爬虫已经把百度网页下所有的页面都抓取出来了:

java爬虫生成word文档 java写爬虫程序_html_03

*

总结思路

1.下载网页我们需要httpclient-3.1.Jar
2.解析网页我们需要htmlparser-1.6.Jar
3.使用htmlparser有时候会报缺tools-1.5.0Jar包错误,解决方法大概是将系统的tools包改了名复制到指定文件夹。(具体操作忘记了,原谅我)。

4.首先要下载网页,我们需要使用httpclient库写一个downTool类,其中使用正则表达去除网页地址多余的字符,以生成清晰的网页文件名来保存。

5.下载网页后,我们要根据网页中的内容读取出需要继续爬的网页地址,所以我们要使用htmlperser解析网页,因此写一个HtmlParserTool类。

6.解析网页得出网页链接的地址后(<a></a><frame></frame>),我们要判断这地址是属于我们要爬的目标网站的,还是属于链接到其它网站的。所以我们要写个LinkFilter过滤器接口,以此判断地址是不是目标网站的。为什么是接口呢?因为我们目标的网站会经常变啊,过滤方法也可能经常变更,如果写死了在类里面,那不就很难更改了。写成接口,需要时才实现就方便很多了。随便写个新的实现就可以改动了。

7.判断完是属于目标网站后,我们又要来判断了。

7.1判断得到的网址我们以前有没有下载过,解析过。如果有就不用重复工作了。所以我们要写集合存储已经下载过的网址,这个数据结构需要不重复并且能快速查找,所以选择HashSet来存储,生成一个集合名字叫作visitedUrl 。

7.2另外我们还要将未下载的网址存到另一个集合,以用来从中提取后下载。考虑到需要先进先出所以采用队列,所以我们用LinkedList来实现一个Queue(队列)接口,以此写了一个Queue类,生成一个集合unVisitedUrl 。

8.最后写一个封装类,调用downTool下载,调用HtmlParserTool来解析网页内容,然后得出新的需要下载的网址,再重新调用downTool。以此循环爬出整个目标网站的数据。

多谢,多谢大家,我是谭咏麟Fands.