Java-网络爬虫(三)

文章目录

  • 前言
  • 一、爬虫的分类
  • 二、跳转页面的爬取
  • 三、网页去重
  • 四、综合案例
    • 1. 案例三


上篇:Java-网络爬虫(二)

前言

上篇文章介绍了 webMagic,通过一个简单的入门案例,对 webMagic 的核心对象和四大组件都做了简要的说明,以下内容则是继续对 webMagic 的讲解


一、爬虫的分类

爬虫按照系统结构和实现技术,大致可以分为以下几种类型:

通用网络爬虫(General Purpose Web Crawler)

也被称为全网爬虫,这种爬虫的爬取目标资源在全互联网中,爬取目标数据巨大。它主要用于为大型搜索引擎和门户网站采集数据。这类爬虫对于爬行速度和存储空间要求较高,对于爬行页面的顺序要求相对较低,同时由于待刷新的页面比较多,通常采用并行工作方式,但需要较长时间才能刷新一次页面,简单的说就是互联网上抓取所有数据。通用网络爬虫的基本构成包括初始 URL 集合、URL 队列、页面爬行模块、页面分析模块、页面数据库、链接过滤模块等。

聚焦网络爬虫(Focused Web Crawler)

也被称为主题网络爬虫,这种爬虫选择性地爬取那些与预先定义好的主题相关的页面。聚焦网络爬虫的目标是只抓取互联网上某一种数据。和通用网络爬虫相比,聚焦爬虫只需要爬行与主题相关的页面,极大地节省了硬件和网络资源,保存的页面也由于少量而更新快,还可以很好地满足一些特定人群对特定领域信息的需求。

量式网络爬虫(Incremental Web Crawler)

这种爬虫会不断爬取数据,但仅爬取新产生的或者已经发生变化的网页,它能够在一定程度上保证所爬行的页面是尽可能新的页面。和周期性爬行和刷新页面的网络爬虫相比,增量式爬虫只会在需要的时候爬行新产生或发生更新的页面,并不重新下载没有发生变化的页面,可有效减少数据下载量,及时更新已爬行的网页,减小时间和空间上的耗费,但是增加了爬行算法的复杂度和实现难度。

深层网络爬虫(Deep Web Crawler)

也被称为深网爬虫,这种爬虫主要抓取隐藏在搜索表单后的、不能通过静态链接获取的网页。这些页面只有当用户提交一些关键词才能获得。


二、跳转页面的爬取

在很多情况下,当我们爬取一个页面的信息时,要通过一些链接进入到其它的页面,进而继续爬取更多的信息

在这里插入图片描述

从上文介绍 webMagic 的原理中可以知道,只需要将待处理的 Request 放入 Scheduler 中就行了,Spider 会从 Scheduler 拉取 Requset 进行抓取,可以通过 page.addTargetRequests(Iterable<String> requests) 实现这一步,代码如下:

    @Overridepublic void process(Page page) {// 解析处理...// 待抓取的 URLList<String> waitUrls = new ArrayList<>();waitUrls.add("url_1");waitUrls.add("url_2");waitUrls.add("url_3");// 将待抓取的 URL 追加到 targetRequests 中page.addTargetRequests(waitUrls);// 持久化处理...}

同样我们可以去追寻源码,进入到 addTargetRequests(Iterable<String> requests) 方法中:

在这里插入图片描述

好像也没有看到将 Request 放入到 Scheduler 中,再看回 spider.run() 方法:

在这里插入图片描述

进入到 processRequest(Request request) 方法:

在这里插入图片描述

从上述源码可知,在运行完 process() 方法后会进入到一个 extractAndAddRequests() 方法来添加一些额外的 Requests,而这些 Requests 其实就是前面通过 page.addTargetRequests() 添加进去的,不妨再看看 extractAndAddRequests() 这个方法

在这里插入图片描述

到这里就可以看到通过 page.addTargetRequests() 这个方法确实会将待处理的 URL 全部推送至 Scheduler


三、网页去重

倘若出现了这么一种场景,在页面 web_1 中有跳转到页面 web_2URL,而在网页 web_2 中也存在着跳转到网页 web_1URL

在这里插入图片描述

那么就会出现这样一种现象,就是 web_1web_2 这两个页面会被无休止的重复解析,这显然是不合理的,所以我们需要记录下已被解析过的页面,让其不能重复解析,这就是页面的去重

对于页面的去重,可以采取很多种方式,这里我就列举三种常用的方法:

  • HashSet 去重:使用 Java 中的 HashSet 不能重复的特点进行去重
    • 优点:容易理解,使用方便
    • 缺点:占用内存大,性能较低
  • Redis 去重:使用 redis 的 set 进行去重
    • 优点:速度快,而且去重不会占用爬虫服务器的资源,可以处理更大数据量的数据爬取
    • 缺点:需要准备 redis 服务器,成本增加
  • 布隆过滤器(BloomFilter):使用布隆过滤器的特性进行去重
    • 优点:相比于 HashSet 去重更快,更加节省内存,也适用于大量数据的去重
    • 缺点:有误判的可能,没有重复可能会被判断为重复,但是重复数据一定会判定重复

关于布隆过滤器的原理可参见博客:Java-布隆过滤器的实现

实际上 webbMagic 提供的 Scheduler 组件已经帮我们解决了上述问题,Scheduler 不仅会将待抓取的 URL 放到队列中进行管理,还会对比已抓取的 URL 进行去重

webMagic 内置了几个常用的 Scheduler,如果只是本地执行规模比较小的爬虫,基本无需定制 Scheduler

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

去重部分被单独抽象成了一个接口 — DuplicateRemover

public interface DuplicateRemover {boolean isDuplicate(Request var1, Task var2);void resetDuplicateCheck(Task var1);int getTotalRequestsCount(Task var1);
}

从而可以为同一个 Scheduler 选择不同的去重方式,以适配不同的需要,目前提供了两种去重方式:

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

除了 RedisSchedulerRedisPriorityScheduler 是使用 redisset 进行去重,其它的 Scheduler 默认使用 HashSetDuplicateRemover 进行去重的

在这里插入图片描述
我们可以通过类图类验证,DuplicateRemovedScheduler 是其它的 Scheduler 的基类,在 DuplicateRemovedScheduler 中使用的就是 HashSetDuplicateRemover,所以其它的 Scheduler 也继承了这一点

在这里插入图片描述

但是 RedisScheduler 实现了 DuplicateRemover 接口,重写了其中的去重逻辑,使用 set 来存储 url,而 RedisPriorityScheduler 又继承了 RedisScheduler,所以这两个是使用了 redisset 进行去重的

在这里插入图片描述

如果需要使用到布隆过滤器进行去重,则需要进行设置

在设置之前还需要先添加 guava 依赖,因为 webMagic-extension 中使用的布隆过滤器是 guava 中的 BloomFilter

maven 导入 guava 依赖

<!-- guava -->
<dependency><groupId>com.google.guava</groupId><artifactId>guava</artifactId><version>33.0.0-jre</version>
</dependency>

代码设置 Scheduler 使用布隆过滤器

        // 创建 schedulerQueueScheduler scheduler = new QueueScheduler();// 设置 scheduler 使用布隆过滤器,预计存放一百万条数据scheduler.setDuplicateRemover(new BloomFilterDuplicateRemover(1000000));// Spider 设置 schedulerspider.setScheduler(scheduler);

四、综合案例

1. 案例三

下载食品营养成分查询平台的所有页面持久化到本地磁盘中

在这里插入图片描述

分析:

  • ① 如果需要爬到该网站的所有页面,就需要将每个页面的超链接放入待处理 URL
  • ② 对网页进行去重处理,可以使用布隆过滤器
  • ③ 要将网页内容持久化到本地磁盘可以使用 FilePipeline

代码如下:

import com.google.common.base.Charsets;
import org.apache.commons.lang3.StringUtils;
import org.jsoup.nodes.Document;
import org.jsoup.nodes.Element;
import org.jsoup.select.Elements;
import us.codecraft.webmagic.Page;
import us.codecraft.webmagic.Site;
import us.codecraft.webmagic.Spider;
import us.codecraft.webmagic.pipeline.FilePipeline;
import us.codecraft.webmagic.processor.PageProcessor;
import us.codecraft.webmagic.scheduler.BloomFilterDuplicateRemover;
import us.codecraft.webmagic.scheduler.QueueScheduler;
import us.codecraft.webmagic.selector.Html;
import java.util.Collections;
import java.util.List;public class WebMagicDemo01 implements PageProcessor {@Overridepublic void process(Page page) {// 获取当前页面的 Html 对象Html html = page.getHtml();// 获取页面上所有的链接List<String> links = html.links().all();// 放入待处理 url 中page.addTargetRequests(links);// 将 html 内容放置 resultItems 中page.putField("html", html.get());}@Overridepublic Site getSite() {// 返回自定义 Sitereturn Site.me()// 设置字符集.setCharset(Charsets.UTF_8.name())// 设置超时时间:5000(单位毫秒).setTimeOut(5000)// 设置重试间隔时间:3000(单位毫秒).setRetrySleepTime(3000)// 设置重试次数:5.setRetryTimes(5);}public static void main(String[] args) {// 创建 spiderSpider spider = Spider.create(new WebMagicDemo01());// 创建 schedulerQueueScheduler scheduler = new QueueScheduler();// 设置 scheduler 使用布隆过滤器,预计存放一百万条数据scheduler.setDuplicateRemover(new BloomFilterDuplicateRemover(1000000));// Spider 设置 schedulerspider.setScheduler(scheduler);// 创建 filePipelineFilePipeline filePipeline = new FilePipeline();// 设置存放路径filePipeline.setPath("D:\\web-magic\\download-page");// Spider 设置 pipelinespider.setPipelines(Collections.singletonList(filePipeline));// 设置初始 URLspider.addUrl("http://yycx.yybq.net/");// 开启 2 个线程spider.thread(2);// 异步爬取spider.runAsync();}
}

可以看到指定文件夹下就开始下载这个网站的页面文件了

在这里插入图片描述

将下载的 html 文件内容和网页源码对比,可以看见基本上是一致的,不过多了一点内容

在这里插入图片描述

在游览器上打开下载的 html 文件

在这里插入图片描述

可以看到虽然下载的 html 文件中的内容和网页源代码几乎一样,但是却没有样式,图片也显示不了,链接跳转过去也是 404,原因是在源码中有关资源的路径使用的都是相对路径,而本地没有这些资源当然访问不了

如果想要解决上述问题:

  • ① 去除多余的内容
  • ② 游览器打开下载的 html 文件,资源部分未能加载

解决方案:

  • 问题 ①:重写 FilePipeline 去除打印多余部分的代码
  • 问题 ②:将有关资源的路径前全部加上起始网页地址

改进后代码:

import com.google.common.base.Charsets;
import org.apache.commons.codec.digest.DigestUtils;
import org.apache.commons.lang3.StringUtils;
import org.jsoup.nodes.Document;
import org.jsoup.nodes.Element;
import org.jsoup.select.Elements;
import us.codecraft.webmagic.*;
import us.codecraft.webmagic.pipeline.FilePipeline;
import us.codecraft.webmagic.processor.PageProcessor;
import us.codecraft.webmagic.scheduler.BloomFilterDuplicateRemover;
import us.codecraft.webmagic.scheduler.QueueScheduler;
import us.codecraft.webmagic.selector.Html;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Map;public class WebMagicDemo01 implements PageProcessor {@Overridepublic void process(Page page) {// 获取当前页面的 Html 对象Html html = page.getHtml();// 获取页面上所有的链接List<String> links = html.links().all();// 放入待处理 url 中page.addTargetRequests(links);// 获取所有的 src 属性Document document = html.getDocument();Elements srcElements = document.select("script,img");for (Element element : srcElements) {if (StringUtils.isNotBlank(element.baseUri())) {String src = element.attr("abs:src");// 重新设置属性element.attr("src", src);}}// 获取所有的标签属性Elements aElements = document.select("a,link");for (Element element : aElements) {if (StringUtils.isNotBlank(element.baseUri())) {String href = element.attr("abs:href");// 重新设置属性element.attr("href", href);}}// 将 html 内容放置 resultItems 中page.putField("html", html.get());}@Overridepublic Site getSite() {// 返回自定义 Sitereturn Site.me()// 设置字符集.setCharset(Charsets.UTF_8.name())// 设置超时时间:5000(单位毫秒).setTimeOut(5000)// 设置重试间隔时间:3000(单位毫秒).setRetrySleepTime(3000)// 设置重试次数:5.setRetryTimes(5);}public static void main(String[] args) {// 创建 spiderSpider spider = Spider.create(new WebMagicDemo01());// 创建 schedulerQueueScheduler scheduler = new QueueScheduler();// 设置 scheduler 使用布隆过滤器,预计存放一百万条数据scheduler.setDuplicateRemover(new BloomFilterDuplicateRemover(1000000));// Spider 设置 schedulerspider.setScheduler(scheduler);// 创建 filePipelineFilePipeline filePipeline = new MyFilePipeline();// 设置存放路径filePipeline.setPath("D:\\web-magic\\download-page");// Spider 设置 pipelinespider.setPipelines(Collections.singletonList(filePipeline));// 设置初始 URLspider.addUrl("http://yycx.yybq.net/");// 开启 2 个线程spider.thread(2);// 异步爬取spider.runAsync();}
}/*** 继承 FilePipeline*/
class MyFilePipeline extends FilePipeline {/*** 重写 FilePipeline 中的 process 方法* 去除打印多余部分的代码*/@SuppressWarnings("all")public void process(ResultItems resultItems, Task task) {String path = this.path + PATH_SEPERATOR + task.getUUID() + PATH_SEPERATOR;try {PrintWriter printWriter = new PrintWriter(new OutputStreamWriter(new FileOutputStream(this.getFile(path + DigestUtils.md5Hex(resultItems.getRequest().getUrl()) + ".html")), "UTF-8"));Iterator var5 = resultItems.getAll().entrySet().iterator();while(true) {while(var5.hasNext()) {Map.Entry<String, Object> entry = (Map.Entry)var5.next();if (entry.getValue() instanceof Iterable) {Iterable value = (Iterable)entry.getValue();Iterator var8 = value.iterator();while(var8.hasNext()) {Object o = var8.next();printWriter.println(o);}} else {printWriter.println((String)entry.getValue());}}printWriter.close();break;}} catch (IOException var10) {var10.printStackTrace();}}
}

再使用游览器打开新下载好的 html 文件就会能看到,其样式也官方网站的效果一样了

在这里插入图片描述

PS:以上案例只做学习爬虫使用,切勿恶意攻击他人网站

上篇:Java-网络爬虫(二)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/342192.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据聚合、自动补全、数据同步、es集群

目录 数据聚合 聚合的分类 DSL实现bucket聚合 DSL实现Metrics聚合 RestAPI实现聚合 多条件聚合 带过滤条件的聚合 自动补全 安装拼音分词器 自定义分词器 completion suggester查询 修改索引库数据结构 RestAPI实现自动补全查询 实现搜索框自动补全 数据同步 数…

数据库读写分离设计方案

一、什么是读写分离 数据库读写分离是一种架构模式&#xff0c;其中数据库系统被配置为将读操作&#xff08;查询&#xff09;和写操作&#xff08;插入、更新、删除&#xff09;分别路由到不同的数据库实例或节点。读操作通常由从节点&#xff08;或只读节点&#xff09;处理&…

自动化测试框架pytest系列之21个命令行参数介绍(二)

第一篇 &#xff1a; 自动化测试框架pytest系列之基础概念介绍(一)-CSDN博客 接上文 3.pytest功能介绍 3.1 第一条测试用例 首先 &#xff0c;你需要编写一个登录函数&#xff0c;主要是作为被测功能&#xff0c;同时编写一个测试脚本 &#xff0c;进行测试登录功能 。 登…

视频转为序列图的软件,让视频批量转为序列图

你是否曾经遇到过这样的困境&#xff1a;需要将一段视频转为一系列的图片&#xff0c;但却没有合适的工具来完成&#xff1f;或许你曾经手动截图&#xff0c;或者用其他方式&#xff0c;但结果往往不尽如人意&#xff0c;图片质量差、色彩失真、画面不清晰。现在&#xff0c;让…

建议收藏!2023首获中科院分区期刊目录

2023年12月27日&#xff0c;中科院分区正式发布《2023年中国科学院文献情报中心期刊分区表》。今年期刊分区表包括SCIE、SSCI、A&HCI&#xff0c;以及ESCI中国期刊&#xff0c;共设置了包括自然科学、人文科学和社会科学在内的21个大类。 相比之前&#xff0c;中科院分区今…

CAN总线记录仪给出口车辆的应用

CAN总线记录仪给出口车辆的应用 CAN总线记录仪在出口车辆中的应用十分广泛&#xff0c;主要体现在故障排查、数据分析、质量保证和驾驶行为分析等方面。 1.故障排查&#xff1a;CAN总线记录仪可以记录车辆在运行过程中的所有CAN总线数据&#xff0c;包括发动机、刹车系统、转…

【AI大模型应用开发】1.0 Prompt Engineering(提示词工程)- 典型构成、原则与技巧,代码中加入Prompt

从这篇文章开始&#xff0c;我们就正式开始学习AI大模型应用开发的相关知识了。首先是提示词工程&#xff08;Prompt Engineering&#xff09;。 文章目录 0. 什么是提示词&#xff08;Prompt&#xff09;1. 为什么Prompt会起作用 - 大模型工作原理2. Prompt的典型构成、原则与…

聚会喝酒摇色子活跃气氛神器小程序源码系统 附带完整的搭建教程

在中国的传统聚会文化中&#xff0c;喝酒摇色子是一种非常受欢迎的游戏方式。它能迅速拉近人与人之间的距离&#xff0c;使气氛更加活跃。然而&#xff0c;传统的摇色子方式存在很多不便&#xff0c;如需要手动计数、无法记录历史数据等。因此&#xff0c;开发一款聚会喝酒摇色…

mp4文件全部转换为mp3

问题 今天突发奇想&#xff0c;想把mp4视频转换为mp3来收听&#xff0c;于是想到了ffmpeg工具 步骤 安装ffmpeg环境 要在 Windows 上配置 FFmpeg 环境&#xff0c;你可以按照以下步骤进行操作&#xff1a; 下载 FFmpeg&#xff1a; 首先&#xff0c;你需要下载 FFmpeg 的 W…

spaceship

通过数字平台启动您的网站、想法和未来&#xff0c;该平台旨在提供和连接您所需的域、托管、电子邮件和 Web 工具&#xff0c;并让您完全掌控 如果需要购买可以开5347的卡&#xff0c;点击获取

【AI视野·今日NLP 自然语言处理论文速览 第七十四期】Wed, 10 Jan 2024

AI视野今日CS.NLP 自然语言处理论文速览 Wed, 10 Jan 2024 Totally 38 papers &#x1f449;上期速览✈更多精彩请移步主页 Daily Computation and Language Papers Model Editing Can Hurt General Abilities of Large Language Models Authors Jia Chen Gu, Hao Xiang Xu, J…

Linux CentOS 7.6安装nginx详细保姆级教程

一、通过wget下载nginx压缩包 1、进入home文件并创建nginx文件夹用来存放nginx压缩包 cd /home //进入home文件夹 mkdir nginx //创建nginx文件夹 cd nginx //进入nginx文件夹2、下载nginx,我这里下载的是Nginx 1.24.0版本&#xff0c;如果要下载新版本可以去官网进行下载:…