• java网页数据抓取实例


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

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

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

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

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

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

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

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

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

    import java.io.BufferedReader;
    import java.io.InputStream;
    import java.io.InputStreamReader;
    import java.net.HttpURLConnection;
    import java.net.URL;
    import java.util.ArrayList;
    import java.util.List;
    import java.util.regex.Matcher;
    import java.util.regex.Pattern;
    
    public class HTMLPageParser {
        public static void main(String[] args) throws Exception {
            //目的网页URL地址
            getURLInfo("http://www.ic.net.cn/userSite/publicQuote/quotes_list.php","utf-8");
        }
        public static List<Product> getURLInfo(String urlInfo,String charset) throws Exception {
            //读取目的网页URL地址,获取网页源码
            URL url = new URL(urlInfo);
            HttpURLConnection httpUrl = (HttpURLConnection)url.openConnection();
            InputStream is = httpUrl.getInputStream();
            BufferedReader br = new BufferedReader(new InputStreamReader(is,"utf-8"));
            StringBuilder sb = new StringBuilder();
            String line;
            while ((line = br.readLine()) != null) {
                //这里是对链接进行处理
                line = line.replaceAll("</?a[^>]*>", "");
                //这里是对样式进行处理
                line = line.replaceAll("<(\w+)[^>]*>", "<$1>");
                sb.append(line);
            }
            is.close();
            br.close();
            //获得网页源码
            return getDataStructure(sb.toString().trim());
        }
        static Pattern proInfo 
               = Pattern.compile("<div>(.*?)</div>\s*<div>(.*?)</div>\s*<div>(.*?)</div>\s*<div>(.*?)</div>\s*<div>(.*?)</div>", Pattern.DOTALL);
        private static List<Product> getDataStructure(String str) {
            //运用正则表达式对获取的网页源码进行数据匹配,提取我们所要的数据,在以后的过程中,我们可以采用httpclient+jsoup,
            //现在暂时运用正则表达式对数据进行抽取提取
            String[] info = str.split("</li>");
            List<Product> list = new ArrayList<Product>();
            for (String s : info) {
                Matcher m = proInfo.matcher(s);
                Product p = null;
                if (m.find()) {
                    p = new Product();
                    //设置产品型号
                    String[] ss = m.group(1).trim().replace(" ", "").split(">");
                    p.setProStyle(ss[1]);
                    //设置产品数量
                    p.setProAmount(m.group(2).trim().replace(" ", ""));
                    //设置产品报价
                    p.setProPrice(m.group(4).trim().replace(" ", ""));
                    //设置产品供应商
                    p.setProSupplier(m.group(5).trim().replace(" ", ""));
                    list.add(p);
                }
            }
            //这里对集合里面不是我们要提取的数据进行移除
            list.remove(0);
            for (int i = 0; i < list.size(); i++) {
                System.out.println("产品型号:"+list.get(i).getProStyle()+",产品数量:"+list.get(i).getProAmount()
                        +",产品报价:"+list.get(i).getProPrice()+",产品供应商:"+list.get(i).getProSupplier());
            }
            return list;
        }
    }
    class Product {
        private String proStyle;//产品型号
        private String proAmount;//产品数量
        private String proPrice;//产品报价
        private String proSupplier;//产品供应商
        public String getProStyle() {
            return proStyle;
        }
        public void setProStyle(String proStyle) {
            this.proStyle = proStyle;
        }
        public String getProSupplier() {
            return proSupplier;
        }
        public void setProSupplier(String proSupplier) {
            this.proSupplier = proSupplier;
        }
        
        public String getProAmount() {
            return proAmount;
        }
        public void setProAmount(String proAmount) {
            this.proAmount = proAmount;
        }
        public String getProPrice() {
            return proPrice;
        }
        public void setProPrice(String proPrice) {
            this.proPrice = proPrice;
        }
        public Product() {
            
        }
        @Override
        public String toString() {
            return "Product [proAmount=" + proAmount + ", proPrice=" + proPrice
                    + ", proStyle=" + proStyle + ", proSupplier=" + proSupplier
                    + "]";
        }
        
    }
    

      

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

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

    我们匹配的数据可能在项目实际开发中,要求我们对所提取的数据要进行数据储存,方便我们下一次进行数据的查询操作。

  • 相关阅读:
    springmvc文件上传 并读取excel文件基本写法 多文件时参数为 @RequestParam MultipartFile[] myfiles 单文件时直接传File
    谷歌浏览器 js调试方法
    jxl实现文件导入页面例子
    angularjs实现上传文件动态显示文件列表
    文件上传 多个文件上传与单个文件上传
    angularjs实现动态表格的删除与增加
    2017songyunxin
    百万数据导出
    OutProductController
    DownloadUtil
  • 原文地址:https://www.cnblogs.com/firstdream/p/9357085.html
Copyright © 2020-2023  润新知