Java网页数据抓取实例

来源:互联网 发布:网络造谣案例 编辑:程序博客网 时间:2024/04/28 22:15

在很多行业中,要对行业数据进行分类汇总,及时分析行业数据,对于公司未来的发展,有很好的参照和横向对比。所以,在实际工作,我们可能要遇到数据采集这个概念,数据采集的最终目的就是要获得数据,提取有用的数据进行数据提取和数据分类汇总。

很多人在第一次了解数据采集的时候,可能无从下手,尤其是作为一个新手,更是感觉很是茫然,所以,在这里分享一下自己的心得,希望和大家一起分享技术,如果有什么不足,还请大家指正。写出这篇目的,就是希望大家一起成长,我也相信技术之间没有高低,只有互补,只有分享,才能使彼此更加成长

在网页数据采集的时候,我们往往要经过这几个大的步骤:

①通过URL地址读取目标网页②获得网页源码③通过网页源码抽取我们要提取的目的数据④对数据进行格式转换,获得我们需要的数据。

这是一个示意图,希望大家了解

了解了基本流程,下来,我以一个案例具体实现如何提取我们需要的数据,对于数据提取可以用正则表达式进行提取,也可以用httpclient+jsoup进行提取,此处,暂且不讲解httpclient+jsou提取网页数据的做法,以后会对httpclient+jsoup进行专门的讲解,此处,先开始讲解如何用正则表达式对数据进行提取。

我在这里找到一个网站:http://www.ic.net.cn/userSite/publicQuote/quotes_list.php 我们要对里面的数据进行提取操作,我们要提取的最终结果是产品的型号、数量、报价、供应商,首先,我们看到这个网站整个页面预览


其次我们看网页源码结构:


上面源码可以很清楚的可以看到整个网页源码结构,下来我们就对整个网页数据进行提取

[java] view plain copy
  1. import java.io.BufferedReader;  
  2. import java.io.InputStream;  
  3. import java.io.InputStreamReader;  
  4. import java.net.HttpURLConnection;  
  5. import java.net.URL;  
  6. import java.util.ArrayList;  
  7. import java.util.List;  
  8. import java.util.regex.Matcher;  
  9. import java.util.regex.Pattern;  
  10.   
  11. public class HTMLPageParser {  
  12.     public static void main(String[] args) throws Exception {  
  13.         //目的网页URL地址  
  14.         getURLInfo("http://www.ic.net.cn/userSite/publicQuote/quotes_list.php","utf-8");  
  15.     }  
  16.     public static List<Product> getURLInfo(String urlInfo,String charset) throws Exception {  
  17.         //读取目的网页URL地址,获取网页源码  
  18.         URL url = new URL(urlInfo);  
  19.         HttpURLConnection httpUrl = (HttpURLConnection)url.openConnection();  
  20.         InputStream is = httpUrl.getInputStream();  
  21.         BufferedReader br = new BufferedReader(new InputStreamReader(is,"utf-8"));  
  22.         StringBuilder sb = new StringBuilder();  
  23.         String line;  
  24.         while ((line = br.readLine()) != null) {  
  25.             //这里是对链接进行处理  
  26.             line = line.replaceAll("</?a[^>]*>""");  
  27.             //这里是对样式进行处理  
  28.             line = line.replaceAll("<(\\w+)[^>]*>""<$1>");  
  29.             sb.append(line);  
  30.         }  
  31.         is.close();  
  32.         br.close();  
  33.         //获得网页源码  
  34.         return getDataStructure(sb.toString().trim());  
  35.     }  
  36.     static Pattern proInfo   
  37.            = Pattern.compile("<div>(.*?)</div>\\s*<div>(.*?)</div>\\s*<div>(.*?)</div>\\s*<div>(.*?)</div>\\s*<div>(.*?)</div>", Pattern.DOTALL);  
  38.     private static List<Product> getDataStructure(String str) {  
  39.         //运用正则表达式对获取的网页源码进行数据匹配,提取我们所要的数据,在以后的过程中,我们可以采用httpclient+jsoup,  
  40.         //现在暂时运用正则表达式对数据进行抽取提取  
  41.         String[] info = str.split("</li>");  
  42.         List<Product> list = new ArrayList<Product>();  
  43.         for (String s : info) {  
  44.             Matcher m = proInfo.matcher(s);  
  45.             Product p = null;  
  46.             if (m.find()) {  
  47.                 p = new Product();  
  48.                 //设置产品型号  
  49.                 String[] ss = m.group(1).trim().replace(" """).split(">");  
  50.                 p.setProStyle(ss[1]);  
  51.                 //设置产品数量  
  52.                 p.setProAmount(m.group(2).trim().replace(" """));  
  53.                 //设置产品报价  
  54.                 p.setProPrice(m.group(4).trim().replace(" """));  
  55.                 //设置产品供应商  
  56.                 p.setProSupplier(m.group(5).trim().replace(" """));  
  57.                 list.add(p);  
  58.             }  
  59.         }  
  60.         //这里对集合里面不是我们要提取的数据进行移除  
  61.         list.remove(0);  
  62.         for (int i = 0; i < list.size(); i++) {  
  63.             System.out.println("产品型号:"+list.get(i).getProStyle()+",产品数量:"+list.get(i).getProAmount()  
  64.                     +",产品报价:"+list.get(i).getProPrice()+",产品供应商:"+list.get(i).getProSupplier());  
  65.         }  
  66.         return list;  
  67.     }  
  68. }  
  69. class Product {  
  70.     private String proStyle;//产品型号  
  71.     private String proAmount;//产品数量  
  72.     private String proPrice;//产品报价  
  73.     private String proSupplier;//产品供应商  
  74.     public String getProStyle() {  
  75.         return proStyle;  
  76.     }  
  77.     public void setProStyle(String proStyle) {  
  78.         this.proStyle = proStyle;  
  79.     }  
  80.     public String getProSupplier() {  
  81.         return proSupplier;  
  82.     }  
  83.     public void setProSupplier(String proSupplier) {  
  84.         this.proSupplier = proSupplier;  
  85.     }  
  86.       
  87.     public String getProAmount() {  
  88.         return proAmount;  
  89.     }  
  90.     public void setProAmount(String proAmount) {  
  91.         this.proAmount = proAmount;  
  92.     }  
  93.     public String getProPrice() {  
  94.         return proPrice;  
  95.     }  
  96.     public void setProPrice(String proPrice) {  
  97.         this.proPrice = proPrice;  
  98.     }  
  99.     public Product() {  
  100.           
  101.     }  
  102.     @Override  
  103.     public String toString() {  
  104.         return "Product [proAmount=" + proAmount + ", proPrice=" + proPrice  
  105.                 + ", proStyle=" + proStyle + ", proSupplier=" + proSupplier  
  106.                 + "]";  
  107.     }  
  108.       
  109. }  

好了,运行上面程序,我们得到下面的数据,就是我们要获得的最终数据




获得数据成功,这就是我们要获得最终的数据结果,最后我要说的是,此处这个网页算是比较简单的,而且,网页源码可以看到源数据,并且此方式是以get方式进行数据提交,真正采集的时候,有些网页结构比较复杂,可能会存在着源码里面没有我们所要提取的数据,关于这一点的解决方式,以后给大家进行介绍。还有,我在采集这个页面的时候,只是采集了当前页面的数据,它还有分页的数据,关于这个我此处不做讲解,只是提示一点,我们可以采用多线程对所有分页的当前数据进行采集,通过线程一个采集当前页面数据,一个进行翻页动作,就可以采集完所有数据。

我们匹配的数据可能在项目实际开发中,要求我们对所提取的数据要进行数据储存,方便我们下一次进行数据的查询操作,下一节我会给大家分享自己有关如何对采集的数据进行数据库储存。

1 0