本博客属原创文章,欢迎转载!转载请务必注明出处:http://guoyunsky.iteye.com/blog/644396

      本博客已迁移到本人独立博客: http://www.yun5u.com/

       欢迎加入Heritrix群(QQ):109148319,10447185 , Lucene/Solr群(QQ) :  118972724

前面说过Heritrix可以在某个抓取基础上(这里假设为A)继续抓取,因为Heritrix对每一个URL都会有相应的日志处理,同时还有checkpoint(备份中心)。所以只要通过A上的日志就可以完全按照该基础A继续抓取,不会重复抓取任何A抓过的任何东西,也会继续抓取A没有抓过的URL。做到这个有两种方法,一种是针对Web启动的,还有一种是针对我上次写的不通过Web启动的方式(启动方式见博客:Heritrix源码分析(五) 如何让Heritrix在Ecplise等IDE下编程启动).

    1)下面介绍这两种启动方式,第一种,通过Web启动:

进入页面,选择:Jobs->Base on a recovery->然后再选择你要二次抓取的Job中的recover-log->然后再点击按钮Submit Job。之后回到Console页面,点击Start。之后就开始再你想要的基础上抓取了。你可以进入这个新建的抓取JOB下的logs目录,发现里面有个recover.gz大小跟你想要二次抓取JOB中的recover.gz大小一般大。以及你去查看该job下的order.xml中的<string name="recover-path"></string>中间的值发现就是你要二次抓取job下recover.gz的绝对目录

  2)不通过Web方式启动:

这个相对简单,只要修改order.xml中<string name="recover-path">D:/recover.gz</string>,中间的值就是你想二次抓取的JOB中logs目录下recover.gz的绝对路径。

同时最好更新一下值为:
               <boolean name="checkpoint-copy-bdbje-logs">true</boolean>
               <boolean name="recover-retain-failures">false</boolean>
               <boolean name="recover-scope-includes">false</boolean>
               <boolean name="recover-scope-enqueues">false</boolean>

至于为什么要这样设置,请参考我关于order.xml介绍的博客:Heritrix源码分析(二) 配置文件order.xml介绍

同时可能你已经知道某些URL不需要抓取,比如从数据库导出来的,而这些URL你的Heritrix并没有处理过。所以这些外部的URL就无法通过以上两种办法导入Heritrix了。这里我写了个工具类,有两种方式,一种是你将URL都放在文件中,然后通过这个工具类从这个文件中读取这些URL(必须一行一个URL),导入到Heritrix中。还有一种方法是针对数据库的,你只要提供相应的ResultSet以及该URL对应数据库的字段名,也可以导入Heritrix,下面贴上代码:

  1. import java.io.BufferedReader;
  2. import java.io.File;
  3. import java.io.FileInputStream;
  4. import java.io.FileNotFoundException;
  5. import java.io.IOException;
  6. import java.io.InputStream;
  7. import java.io.InputStreamReader;
  8. import java.io.UnsupportedEncodingException;
  9. import java.sql.Connection;
  10. import java.sql.ResultSet;
  11. import java.sql.SQLException;
  12. import java.sql.Statement;
  13. import org.archive.crawler.frontier.RecoveryJournal;
  14. public class UrlToRecoverUtil {
  15. /**
  16. * 从文件中导入URl到recover.gz以便URL不再被抓取
  17. *
  18. * @param sourceFilePath        URL来源文件
  19. * @param sourceFileEncoding    URL来源文件的编码
  20. * @param recoverGzPath         要写到的recover.gz文件目录
  21. * @param recoverGzFileName     recover.gz文件名,可以为空
  22. * @return
  23. */
  24. public static boolean urlToRecoverUtilByFile(String sourceFilePath,String sourceFileEncoding,String recoverGzDir,String recoverGzFileName){
  25. boolean result=false;
  26. InputStream is=null;
  27. InputStreamReader isr=null;
  28. BufferedReader br=null;
  29. File sourceFile=null;
  30. String line=null;
  31. RecoveryJournal recover = null;
  32. try {
  33. sourceFile=new File(sourceFilePath);
  34. //recover.gz文件为空则采用默认名字
  35. if(recoverGzFileName==null||recoverGzFileName.equals("")){
  36. recoverGzFileName="recover.gz";
  37. }
  38. recover=new RecoveryJournal(recoverGzDir,recoverGzFileName);//构造recover.gz对象
  39. //读取文件内容
  40. is=new FileInputStream(sourceFile);
  41. isr=new InputStreamReader(is,sourceFileEncoding);
  42. br=new BufferedReader(isr);
  43. //一行一行写入recover.gz文件
  44. while((line=br.readLine().trim())!=null){
  45. if(!line.equals("")){
  46. recover.writeLine(RecoveryJournal.F_SUCCESS, line);
  47. }
  48. }
  49. result=true;
  50. } catch (FileNotFoundException e) {
  51. e.printStackTrace();
  52. } catch (UnsupportedEncodingException e) {
  53. e.printStackTrace();
  54. } catch (IOException e) {
  55. e.printStackTrace();
  56. }finally{
  57. try {
  58. if(recover!=null){
  59. recover.close();
  60. }
  61. if(br!=null){
  62. br.close();
  63. }
  64. if(isr!=null){
  65. isr.close();
  66. }
  67. if(is!=null){
  68. is.close();
  69. }
  70. } catch (IOException e) {
  71. e.printStackTrace();
  72. }
  73. }
  74. return result;
  75. }
  76. /**
  77. * 从ResultSet结果集中获取URL导入到recover.gz以便URl不再被抓取
  78. *
  79. * @param rs                ResultSet结果集
  80. * @param filedName         ResultSet结果集中要获取URL对应的字段名
  81. * @param recoverGzDir      要写到的recover.gz文件目录
  82. * @param recoverGzFileName recover.gz文件名,可以为空
  83. * @return
  84. */
  85. public static boolean urlToRecoverUtilByResultSet(ResultSet rs,String filedName,String recoverGzDir,String recoverGzFileName){
  86. boolean result=false;
  87. String line=null;
  88. RecoveryJournal recover = null;
  89. try {
  90. if(recoverGzFileName==null||recoverGzFileName.equals("")){
  91. recoverGzFileName="recover.gz";
  92. }
  93. recover=new RecoveryJournal(recoverGzDir,recoverGzFileName);
  94. if(rs!=null){
  95. while(rs.next()){
  96. line=rs.getString(filedName).trim();
  97. if(!line.equals("")){
  98. recover.writeLine(RecoveryJournal.F_SUCCESS, line);
  99. }
  100. }
  101. result=true;
  102. }
  103. } catch (SQLException e) {
  104. e.printStackTrace();
  105. } catch (IOException e) {
  106. e.printStackTrace();
  107. }finally{
  108. try {
  109. if(rs!=null){
  110. rs.close();
  111. }
  112. if(recover!=null){
  113. recover.close();
  114. }
  115. } catch (SQLException e) {
  116. e.printStackTrace();
  117. }
  118. }
  119. return result;
  120. }
  121. /**
  122. * @param args
  123. */
  124. public static void main(String[] args) {
  125. /*
  126. * 示例,从结果集中写入URL到recover.gz
  127. */
  128. Connection con=null;
  129. Statement stmt=null;
  130. ResultSet rs=null;
  131. String sql="SELECT CLASSIFY_INFO_URL FROM CLASSIFY_INFO";
  132. boolean result=false;
  133. try {
  134. con=DatabaseUtil.getConnection_Mysql_CrawlServer_Local();//获取Connection
  135. stmt=con.createStatement();
  136. rs=stmt.executeQuery(sql);
  137. result=urlToRecoverUtilByResultSet(rs,"CLASSIFY_INFO_URL","D:/HeritrixRecover/",null);
  138. System.out.println("从结果集中导入URL到recover.gz文件:"+(result?"成功!":"失败!"));
  139. } catch (SQLException e) {
  140. // TODO Auto-generated catch block
  141. e.printStackTrace();
  142. }finally{
  143. DatabaseUtil.closeConnection(con, stmt, rs);//关闭Connection、Statement、ResultSet
  144. }
  145. }
  146. }

这个工具类其实主要也只是生成recover.gz文件。如果你采用Web方式启动,你只要找到一个你抓取过的JOB,然后用这个生成的recover.gz目录覆盖你找到job下logs目录中的recover.gz即可。而如果你采用非Web启动就更简单了,只要将order.xml中<string name="recover-path">D:/recover.gz</string>中的值改成你生成recover.gz绝对路径即可!

Heritrix源码分析(九) Heritrix的二次抓取以及如何让Heritrix抓取你不想抓取的URL的更多相关文章

  1. Heritrix源码分析(十四) 如何让Heritrix不间断的抓取(转)

    欢迎加入Heritrix群(QQ):109148319,10447185 , Lucene/Solr群(QQ) :  118972724 本博客已迁移到本人独立博客: http://www.yun5u ...

  2. Heritrix源码分析(十四)

    近段时间在搞定Lucene的一些问题,所以Heritrix源码分析暂时告一段落.今天下午在群里有同学提到了Heritrix异常终止的问题以及让Heritrix不停的抓取(就是抓完一遍后载入种子继续抓取 ...

  3. 手机自动化测试:appium源码分析之bootstrap十二

    手机自动化测试:appium源码分析之bootstrap十二   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣 ...

  4. HDFS源码分析之UnderReplicatedBlocks(二)

    UnderReplicatedBlocks还提供了一个数据块迭代器BlockIterator,用于遍历其中的数据块.它是UnderReplicatedBlocks的内部类,有三个成员变量,如下: // ...

  5. 【集合框架】JDK1.8源码分析之LinkedHashMap(二)

    一.前言 前面我们已经分析了HashMap的源码,已经知道了HashMap可以用在哪种场合,如果这样一种情形,我们需要按照元素插入的顺序来访问元素,此时,LinkedHashMap就派上用场了,它保存 ...

  6. 【JUC】JDK1.8源码分析之AbstractQueuedSynchronizer(二)

    一.前言 在锁框架中,AbstractQueuedSynchronizer抽象类可以毫不夸张的说,占据着核心地位,它提供了一个基于FIFO队列,可以用于构建锁或者其他相关同步装置的基础框架.所以很有必 ...

  7. 【JUC】JDK1.8源码分析之ConcurrentSkipListMap(二)

    一.前言 最近在做项目的同时也在修复之前项目的一些Bug,所以忙得没有时间看源代码,今天都完成得差不多了,所以又开始源码分析之路,也着笔记录下ConcurrentSkipListMap的源码的分析过程 ...

  8. 【Zookeeper】源码分析之网络通信(二)

    一.前言 前面介绍了ServerCnxn,下面开始学习NIOServerCnxn. 二.NIOServerCnxn源码分析 2.1 类的继承关系 public class NIOServerCnxn ...

  9. 【Zookeeper】源码分析之服务器(二)

    一.前言 前面阐述了服务器的总体框架,下面来分析服务器的所有父类ZooKeeperServer. 二.ZooKeeperServer源码分析 2.1 类的继承关系 public class ZooKe ...

随机推荐

  1. URAL 1183 Brackets Sequence(DP)

    题目链接 题意 : 给你一串由括号组成的串,让你添加最少的括号使该串匹配. 思路 : 黑书上的DP.dp[i][j] = min{dp[i+1][j-1] (sh[i] == sh[j]),dp[i] ...

  2. (8)nehe教程2-多边形

    参考自: http://www.yakergong.net/nehe/ 你的第一个多边形: 在第一个教程的基础上,我们添加了一个三角形和一个四边形.也许你认为这很简单,但你已经迈出了一大步,要知道任何 ...

  3. eclipse创建python项目

    http://jingyan.baidu.com/article/19192ad8173300e53f570757.html

  4. C语言,一个彩票摇奖程序摇出22选5的中奖号码

    摇奖机摇奖,无非就是利用它的随机性,让球从摇奖机中随机地掉出,就成了中奖号码.而C语言中也同样有个rand()函数可以产生随机数,利用这个rand()函数产生的随机数,同样可以代替从摇奖机中随机摇出的 ...

  5. JavaPersistenceWithHibernate第二版笔记-第四章-Mapping persistent classes-003映射实体时的可选操作(<delimited-identifiers/>、PhysicalNamingStrategy、PhysicalNamingStrategyStandardImpl、、、)

    一.自定义映射的表名 1. @Entity @Table(name = "USERS") public class User implements Serializable { / ...

  6. Struts2笔记——通配符和动态方法调用

     通配符映射 * 一个 Web应用可能有成百上千个 action 声明. 可以利用 struts提供的通配符映射机制把多个彼此相似的映射关系简化为一个映射关系 * 通配符映射规则     > 若 ...

  7. sublime3 乱码问题

    解决方法: 一.安装Package Control 二.按Ctrl+Shift+P打开命令行,输入Install Package,回车,然后继续输入ConvertToUTF8,回车  (把GB2312 ...

  8. 如何在Java客户端调用RESTful服务

    在这个例子中,我们将看到如何使用java.net包实用工具,创建一个访问REST服务RESTful的客户端.当然这不是创建一个RESTful客户端最简单的方法,因为你必须自己读取服务器端的响应,以及J ...

  9. 284. Peeking Iterator

    题目: Given an Iterator class interface with methods: next() and hasNext(), design and implement a Pee ...

  10. linux中U盘umonut时出现“Device is busy”的解决方法

    问题: #umount /dev/sda1 umount: /mnt/usb: device is busy 查找占用目录进程: #lsof |grep /mnt/usb bash 1971 root ...