zoukankan      html  css  js  c++  java
  • WebMagic爬虫初探

    WebMagic初探

    先放上官网,更多资料可自行查阅:http://webmagic.io/docs/zh/

    大致上WebMagic可以分为[Downloader、PageProcessor、Scheduler、Pipeline]四大组件,最外由Spider协调。可以灵活的定制组件功能,我们一次从0到1逐步分析这四大组件

    写在前面:本篇目前只分析了爬取基本页面,至于前端渲染页面,后面做补充

    Spider集大成者

    Spiler协调四个组件。除了PageProcessor是在Spider创建的时候已经指定,DownloaderSchedulerPipeline都可以通过Spider的setter方法来进行配置和更改。

    • addPipeline() //设置Pipeline,一个Spider可以有多个Pipeline

    • setDownloader //设置Downloader

    • setScheduler //设置Scheduler

    public static void main(String[] args) {
       Spider.create(new LianjiaProcessor())
              .addUrl("https://cd.lianjia.com/zufang/")
              .thread(5)
              .addPipeline(new MyPipeline())
              .run();
    }

    上面这个main方法,是我们今天Demo的一个启动类,可以看到,最外层由Spider进行启动和管理,上面已经指定了两个插件,分别是 [PageProcessor : new LianjiaProcessor()] ,以及 [Pipeline:new MyPipeline()],下面我开始入门

    定制PageProcessor

    PageProcessor组件可以理解为,这个类基本上包含了爬取一个网站,你需要写的所有代码

    package com.example.webmagic.processor;

    import com.example.webmagic.pipeline.MyPipeline;
    import us.codecraft.webmagic.Page;
    import us.codecraft.webmagic.Site;
    import us.codecraft.webmagic.Spider;
    import us.codecraft.webmagic.processor.PageProcessor;
    import us.codecraft.webmagic.selector.Html;

    import java.util.List;

    public class LianjiaProcessor implements PageProcessor {

       //创建Site对象,设置重试次数以及爬完一个页面另开的间隔时间
       private Site site = Site.me().setRetryTimes(3).setSleepTime(200);

       @Override
       public void process(Page page) {
           //会得到一个html页面
           Html html = page.getHtml();

           //得到所有的房源详情链接,丢入容器,待会儿会依次进行爬取
           List<String> urlList = html.css(".content__list--item--title a").links().all();
           //将所有的房源详情链接添加到爬取任务队列
           page.addTargetRequests(urlList);

           //这里我们使用Xpath语法去命中我们想要的数据
           //标题
           String title = html.xpath("//div[@class='content clear w1150']/p/text()").toString();
           page.putField("title", title);
           //价格
           page.putField("rent", html.xpath("//div[@class='content__aside--title']/span/text()").toString());
           //租赁方式 、户型,如:2室1厅1卫 、朝向
           page.putField("type", html.xpath("//ul[@class='content__aside__list']/allText()").toString());
           //房源基本信息
           page.putField("info", html.xpath("//div[@class='content__article__info']/allText()").toString());
           //图片信息
           page.putField("img", html.xpath("//div[@class='content__article__slide__item']/img").toString());

         //做一个判断,在房源列表页title应该是为null,且根据分页规律,将更多要爬取的页面丢入到容器中
           if (page.getResultItems().get("title") == null) {
               page.setSkip(true);
               //分页
               for (int i = 1; i <= 100; i++) {
                   page.addTargetRequest("https://cd.lianjia.com/zufang/pg" + i);
              }
          }
      }

       @Override
       public Site getSite() {
           return site;
      }

       public static void main(String[] args) {
           Spider.create(new LianjiaProcessor())
                  .addUrl("https://cd.lianjia.com/zufang/")
            //启用五个线程
                  .thread(5)
                  .addPipeline(new MyPipeline())
                  .run();
      }
    }

    XPath使用心得

    刚刚我们在爬取的业务中说道了XPath,下面我们了解一下,我们就以Dmeo中爬取的title为列进行说明

    有时候一个选择器定位不到元素,可以通过 父-->子的嵌套关系进行选择,一般用class做选择,当命中的数据有多条是,可以使用allText()API,获得所有数据,反正边dubug边看嘛,哪个值取不到就看看这个选择器是否有问题

    Scheduler

    前面我们已经看到,我们将所有要爬取的路径抽离出来放到一个 容器中

    //得到所有的房源详情链接
    List<String> urlList = html.css(".content__list--item--title a").links().all();
    //将所有的房源详情链接添加到爬取任务队列
    page.addTargetRequests(urlList);

    Scheduler是WebMagic中进行URL管理的组件。一般来说,Scheduler包括两个作用:

    1. 对待抓取的URL队列进行管理。

    2. 对已抓取的URL进行去重。

    小规模的爬虫工程,无需定制这个组件,使用默认的即可,至于默认的,做了解内容(来自官方)

    说明备注
    DuplicateRemovedScheduler 抽象基类,提供一些模板方法 继承它可以实现自己的功能
    QueueScheduler 使用内存队列保存待抓取URL  
    PriorityScheduler 使用带有优先级的内存队列保存待抓取URL 耗费内存较QueueScheduler更大,但是当设置了request.priority之后,只能使用PriorityScheduler才可使优先级生效
    FileCacheQueueScheduler 使用文件保存抓取URL,可以在关闭程序并下次启动时,从之前抓取到的URL继续抓取 需指定路径,会建立.urls.txt和.cursor.txt两个文件
    RedisScheduler 使用Redis保存抓取队列,可进行多台机器同时合作抓取 需要安装并启动redis

    在0.5.1版本里,作者对Scheduler的内部实现进行了重构,去重部分被单独抽象成了一个接口:DuplicateRemover,从而可以为同一个Scheduler选择不同的去重方式,以适应不同的需要,目前提供了两种去重方式。

    说明
    HashSetDuplicateRemover 使用HashSet来进行去重,占用内存较大
    BloomFilterDuplicateRemover 使用BloomFilter来进行去重,占用内存较小,但是可能漏抓页面

    所有默认的Scheduler都使用HashSetDuplicateRemover来进行去重,(除开RedisScheduler是使用Redis的set进行去重)。如果你的URL较多,使用HashSetDuplicateRemover会比较占用内存,所以也可以尝试以下BloomFilterDuplicateRemover1,使用方式:

    spider.setScheduler(new QueueScheduler()
    .setDuplicateRemover(new BloomFilterDuplicateRemover(10000000)) //10000000是估计的页面数量
    )

    0.6.0版本后,如果使用BloomFilterDuplicateRemover,需要单独引入Guava依赖包

    定制Pipeline

    Pipeline其实就是将PageProcessor抽取的结果,继续进行了处理的部分,之前已经说过一个Spider可以定制多个Pipeline,如下所示就可以实现在控制台打印且执行自定义持久化处理

    Spider.addPipeline(new ConsolePipeline()).addPipeline(new FilePipeline())
    package com.example.webmagic.pipeline;

    import com.fasterxml.jackson.databind.ObjectMapper;
    import org.apache.commons.io.FileUtils;
    import org.apache.commons.io.IOUtils;
    import org.apache.commons.lang3.StringUtils;
    import org.apache.http.client.methods.CloseableHttpResponse;
    import org.apache.http.client.methods.HttpGet;
    import org.apache.http.impl.client.HttpClientBuilder;
    import us.codecraft.webmagic.ResultItems;
    import us.codecraft.webmagic.Task;
    import us.codecraft.webmagic.pipeline.Pipeline;

    import java.io.File;
    import java.util.HashMap;
    import java.util.Map;

    public class MyPipeline implements Pipeline {

       private static final ObjectMapper MAPPER = new ObjectMapper();

       @Override
       public void process(ResultItems resultItems, Task task) {
           Map<String, Object> data = new HashMap<>();
           data.put("url", resultItems.getRequest().getUrl());
           data.put("title", resultItems.get("title"));//标题
           data.put("rent", resultItems.get("rent"));//租金
           String[] types = StringUtils.split(resultItems.get("type"), ' ');
           data.put("rentMethod", types[0]);//租赁方式
           data.put("houseType", types[1]);//户型,如:2室1厅1卫
           data.put("orientation", types[2]);//朝向
           String[] infos = StringUtils.split(resultItems.get("info"), ' ');
           for (String info : infos) {
               if (StringUtils.startsWith(info, "看房:")) {
                //拿到看房信息
                   data.put("time", StringUtils.split(info, ':')[1]);
              } else if (StringUtils.startsWith(info, "楼层:")) {
                //拿到楼层信息
                   data.put("floor", StringUtils.split(info, ':')[1]);
              }
          }
           String imageUrl = StringUtils.split(resultItems.get("img"), '"')[3];
        //图片从命名
           String newName = StringUtils
                  .substringBefore(StringUtils
                          .substringAfterLast(resultItems.getRequest().getUrl(),
                                   "/"), ".") + ".jpg";
           try {
               this.downloadFile(imageUrl, new File("E:\code\images\" + newName));
               data.put("image", newName);
               String json = MAPPER.writeValueAsString(data);
               FileUtils.write(new File("E:\code\data.json"), json + " ", "UTF-8",
                       true);
          } catch (Exception e) {
               e.printStackTrace();
          }
      }
       //图片下载
       public void downloadFile(String url, File dest) throws Exception {
           HttpGet httpGet = new HttpGet(url);
           CloseableHttpResponse response =
                   HttpClientBuilder.create().build().execute(httpGet);
           try {
               FileUtils.writeByteArrayToFile(dest,
                       IOUtils.toByteArray(response.getEntity().getContent()));
          } finally {
               response.close();
          }
      }
    }

    在上面我们的定制Pipeline中,我们将数据持久化到了Json文件,对应图片也保存到本地磁盘,如果是想持久化到第三方数据库,导入依赖进行持久化即可

    Downloader

    WebMagic的默认Downloader基于HttpClient。一般来说,你无须自己实现Downloader,当然这也是官方说明,不过HttpClientDownloader也预留了几个扩展点,以满足不同场景的需求。

    你可能希望通过其他方式来实现页面下载,例如使用SeleniumDownloader来渲染动态页面。

    上面这种方式是爬取基本页面的方式,至于现在JS框架那么多,我们得想法子啊

    爬取渲染页面待补充

    ...

     

  • 相关阅读:
    MySQL启动和关闭命令总结
    MySQL数据库5.6版本首次安装Root密码问题
    tomcat 9性能调优注意事项
    扫除减脂之路上的几个小障碍
    MySQL常见面试题
    关于邮箱发送邮件二之附件及图片
    关于邮箱发送邮件
    关于算法
    python中常见的数据类型
    C++实现复数类的输入输出流以及+-*/的重载
  • 原文地址:https://www.cnblogs.com/msi-chen/p/11311464.html
Copyright © 2011-2022 走看看