又攻下一座山头。

  1. //======================================================
  2. // mzitu图片批量下载爬虫1.00
  3. // 2017年11月19日
  4. //======================================================
  5.  
  6. // 内置https模块
  7. //var https=require("https");
  8.  
  9. // 内置http模块
  10. var http=require("http");
  11.  
  12. // 内置文件处理模块,用于创建目录和图片文件
  13. var fs=require('fs');
  14.  
  15. // 用于转码。非Utf8的网页如gb2132会有乱码问题,需要iconv将其转码
  16. var iconv = require('iconv-lite');
  17.  
  18. // cheerio模块,提供了类似jQuery的功能,用于从HTML code中查找图片地址和下一页
  19. var cheerio = require("cheerio");
  20.  
  21. // 请求参数JSON。http和https都有使用
  22. var options;
  23.  
  24. // request请求
  25. var req;
  26.  
  27. // 图片数组,找到的图片地址会放到这里
  28. var pictures=[];
  29.  
  30. // 存放图片的目录
  31. var folder="";
  32.  
  33. //--------------------------------------
  34. // 爬取网页,找图片地址,再爬
  35. // pageUrl sample:https://www.4493.com/xingganmote/136346/1.htm
  36. // pageUrl sample:
  37. //--------------------------------------
  38. function crawl(pageUrl){
  39. console.log("Current page="+pageUrl);
  40.  
  41. // 得到hostname和path
  42. var currUrl=pageUrl.replace("http://","");
  43. var pos=currUrl.indexOf("/");
  44. var hostname=currUrl.slice(0,pos);
  45. var path=currUrl.slice(pos);
  46. //console.log("hostname="+hostname);
  47. //console.log("path="+path);
  48.  
  49. // 初始化options
  50. options={
  51. hostname:hostname,
  52. port:80,
  53. path:path,// 子路径
  54. method:'GET',
  55. };
  56.  
  57. req=http.request(options,function(resp){
  58. var html = [];
  59.  
  60. resp.on("data", function(data) {
  61. html.push(data);
  62. })
  63. resp.on("end", function() {
  64. var buffer = Buffer.concat(html);
  65. var body = buffer.toString();
  66. //console.log(body);
  67.  
  68. var $ = cheerio.load(body);
  69. var picCount=0;
  70.  
  71. // 找图片放入数组
  72. $(".main-image p a img").each(function(index,element){
  73. var picUrl=$(element).attr("src");
  74. console.log(picUrl);
  75.  
  76. if(picUrl.indexOf('.jpg')!=-1){
  77. pictures.push(picUrl);
  78. picCount++;
  79. }
  80. })
  81. console.log("找到图片"+picCount+"张.");
  82.  
  83. var nextPageUrl=null;
  84. // 找下一页
  85. $(".pagenavi a").each(function(index,element){
  86. var text=$(element).text();
  87. if(text.indexOf('下一页')!=-1){
  88. nextPageUrl=$(element).attr("href");
  89. //nextPageUrl="http://www.mmonly.cc/mmtp/xgmn/"+nextPageUrl;
  90. console.log("找到下一页="+nextPageUrl);
  91. }
  92. })
  93.  
  94. if(nextPageUrl==null){
  95. console.log(pageUrl+"已经是最后一页了.\n");
  96. saveFile(pageUrl,pictures);// 保存
  97. download(pictures);
  98. }else{
  99. console.log("继续下一页");
  100. crawl(nextPageUrl);
  101. }
  102.  
  103. }).on("error", function() {
  104. saveFile(pageUrl,pictures);// 保存
  105. console.log("crawl函数失败,请进入断点续传模式继续进行");
  106. })
  107. });
  108.  
  109. // 超时处理
  110. req.setTimeout(7500,function(){
  111. req.abort();
  112. });
  113.  
  114. // 出错处理
  115. req.on('error',function(err){
  116. console.log('请求发生错误'+err);
  117. saveFile(pageUrl,pictures);// 保存
  118. console.log("crawl函数失败,请进入断点续传模式继续进行");
  119. });
  120.  
  121. // 请求结束
  122. req.end();
  123. }
  124.  
  125. //--------------------------------------
  126. // 下载图片
  127. //--------------------------------------
  128. function download(pictures){
  129.  
  130. var total=0;
  131. total=pictures.length;
  132. console.log("总计有"+total+"张图片将被下载.");
  133. appendToLogfile(folder,"总计有"+total+"张图片将被下载.\n");
  134. for(var i=0;i<pictures.length;i++){
  135. var picUrl=pictures[i];
  136. downloadPic(picUrl,folder);
  137. }
  138. }
  139.  
  140. //--------------------------------------
  141. // 写log文件
  142. //--------------------------------------
  143. function appendToLogfile(folder,text){
  144. fs.appendFile('./'+folder+'/log.txt', text, function (err) {
  145. if(err){
  146. console.log("不能书写log文件");
  147. console.log(err);
  148. }
  149. });
  150. }
  151.  
  152. //--------------------------------------
  153. // 取得当前时间
  154. //--------------------------------------
  155. function getNowFormatDate() {
  156. var date = new Date();
  157. var seperator1 = "-";
  158. var seperator2 = "_";
  159. var month = date.getMonth() + 1;
  160. var strDate = date.getDate();
  161. if (month >= 1 && month <= 9) {
  162. month = "0" + month;
  163. }
  164. if (strDate >= 0 && strDate <= 9) {
  165. strDate = "0" + strDate;
  166. }
  167. var currentdate =date.getFullYear() + seperator1 + month + seperator1 + strDate
  168. + " " + date.getHours() + seperator2 + date.getMinutes()
  169. + seperator2 + date.getSeconds();
  170. return currentdate;
  171. }
  172.  
  173. //--------------------------------------
  174. // 下载单张图片
  175. // picUrl sample:http://i.meizitu.net/2017/11/18b01.jpg
  176. //--------------------------------------
  177. function downloadPic(picUrl,folder){
  178. console.log("图片:"+picUrl+"下载开始");
  179.  
  180. // 得到hostname,path和port
  181. var currUrl=picUrl.replace("http://","");
  182. var pos=currUrl.indexOf("/");
  183. var hostname=currUrl.slice(0,pos);
  184. var path=currUrl.slice(pos);
  185.  
  186. // 有端口加端口,没有端口默认80
  187. var port=80;
  188.  
  189. //console.log("hostname="+hostname);
  190. //console.log("path="+path);
  191. //console.log("port="+port);
  192.  
  193. var picName=currUrl.slice(currUrl.lastIndexOf("/"));
  194.  
  195. // 初始化options
  196. options={
  197. hostname:hostname,
  198. port:port,
  199. path:path,
  200. method:'GET',
  201. headers:{
  202. 'Referer':'http://i.meizitu.net',
  203. },
  204. };
  205.  
  206. req=http.request(options,function(resp){
  207. var imgData = "";
  208. resp.setEncoding("binary");
  209.  
  210. resp.on('data',function(chunk){
  211. imgData+=chunk;
  212. });
  213.  
  214. resp.on('end',function(){
  215.  
  216. // 创建文件
  217. var fileName="./"+folder+picName;
  218. fs.writeFile(fileName, imgData, "binary", function(err){
  219. if(err){
  220. console.log("[downloadPic]文件 "+fileName+" 下载失败.");
  221. console.log(err);
  222. appendToLogfile(folder,"文件 "+picUrl+" 下载失败.\n");
  223. }else{
  224. appendToLogfile(folder,"文件 "+picUrl+" 下载成功.\n");
  225. console.log("文件"+fileName+"下载成功");
  226. }
  227. });
  228. });
  229. });
  230.  
  231. // 超时处理
  232. req.setTimeout(7500,function(){
  233. req.abort();
  234. });
  235.  
  236. // 出错处理
  237. req.on('error',function(err){
  238. if(err){
  239. console.log('[downloadPic]文件 '+picUrl+" 下载失败,"+'因为'+err);
  240. appendToLogfile(folder,"文件"+picUrl+"下载失败.\n");
  241. }
  242. });
  243.  
  244. // 请求结束
  245. req.end();
  246. }
  247.  
  248. //--------------------------------------
  249. // 程序入口
  250. //--------------------------------------
  251. function getInput(){
  252. process.stdin.resume();
  253. process.stdout.write("\033[33m 新建模式输入第一页URL,断点续传模式输入0,请输入: \033[39m");// 草黄色
  254. process.stdin.setEncoding('utf8');
  255.  
  256. process.stdin.on('data',function(text){
  257. var input=text.trim();
  258. process.stdin.end();// 退出输入状态
  259.  
  260. if(text.trim()=='0'){
  261. process.stdout.write("\033[36m 进入断点续传模式. \033[39m"); // 蓝绿色
  262.  
  263. // Read File
  264. fs.readFile('./save.dat','utf8',function(err,data){
  265. if(err){
  266. console.log('读取文件save.dat失败,因为'+err);
  267. }else{
  268. //console.log(data);
  269. var obj=JSON.parse(data);
  270.  
  271. pictures=obj.pictures;
  272. console.log('提取图片'+pictures.length+'张');
  273.  
  274. folder=obj.folder;
  275.  
  276. // 创建目录
  277. fs.mkdir('./'+folder,function(err){
  278. if(err){
  279. console.log("目录"+folder+"已经存在");
  280. }
  281. });
  282.  
  283. crawl(obj.url);
  284. }
  285. });
  286.  
  287. // Resume crawl
  288. }else{
  289. process.stdout.write("\033[35m 进入新建模式. \033[039m"); //紫色
  290.  
  291. folder='pictures('+getNowFormatDate()+")";
  292. // 创建目录
  293. fs.mkdir('./'+folder,function(err){
  294. if(err){
  295. console.log("目录"+folder+"已经存在");
  296. }
  297. });
  298.  
  299. crawl(input);
  300. }
  301. });
  302. }
  303.  
  304. //--------------------------------------
  305. // 将爬行中信息存入数据文件
  306. //--------------------------------------
  307. function saveFile(url,pictures){
  308. var obj=new Object;
  309. obj.url=url;
  310. obj.pictures=pictures;
  311. obj.folder=folder;
  312. var text=JSON.stringify(obj);
  313.  
  314. fs.writeFile('./save.dat',text,function(err){
  315. if(err){
  316. console.log('写入文件save.dat失败,因为'+err);
  317. }
  318. });
  319. }
  320.  
  321. // 调用getInput函数,程序开始
  322. getInput();

2017年11月20日07:43:50

Node.js mzitu图片批量下载爬虫1.00的更多相关文章

  1. Node.js 4493图片批量下载爬虫1.00

    这个爬虫依然需要iconv转码,想不到如今非utf8的网页还这么多.另外此网页找下一页的方式比较异常,又再次借助了正则表达式. 代码如下: //============================ ...

  2. Node.js monly图片批量下载爬虫1.00

    此爬虫又用到了iconv转码,代码如下: //====================================================== // mmonly图片批量下载爬虫1.00 ...

  3. Node.js m03122图片批量下载爬虫1.00

    //====================================================== // m03122图片批量下载爬虫1.00 // 2017年11月18日 //==== ...

  4. Node.js mm131图片批量下载爬虫1.00 iconv协助转码

    //====================================================== // mm131图片批量下载爬虫1.00 // 2017年11月15日 //===== ...

  5. Node.js mimimn图片批量下载爬虫 1.00

    这个爬虫在Referer设置上和其它爬虫相比有特殊性.代码: //====================================================== // mimimn图片批 ...

  6. Node.js nvshens图片批量下载爬虫 1.00

    //====================================================== // www.nvshens.com图片批量下载Node.js爬虫1.00 // 此程 ...

  7. Node.js mm131图片批量下载爬虫1.01 增加断点续传功能

    这里的断点续传不是文件下载时的断点续传,而是指在爬行页面时有时会遇到各种网络中断而从中断前的页面及其数据继续爬行的过程,这个过程和断点续传原理上相似故以此命名.我的具体做法是:在下载出现故障或是图片已 ...

  8. Node.js abaike图片批量下载爬虫1.02

    //====================================================== // abaike图片批量下载爬虫1.02 // 用最近的断点续传框架改写原有1.01 ...

  9. Node.js nvshens图片批量下载爬虫1.01

    //====================================================== // nvshens图片批量下载爬虫1.01 // 用最近的断点续传框架改写原有1.0 ...

随机推荐

  1. KMS使用CLion作为IDE来调试

    KMS使用CLion作为IDE来调试,打开kms相应模块的目录,CLion自动识别相应的CMakeLists.txt. 然而会make失败,经搜索,看到Clion使用的自带的cmake,因此系统中安装 ...

  2. [onethink ucenter] 跨域名单点登录关键点

    1.uc_client/data/cache/apps.php <?php $_CACHE['apps'] = array ( 1 => array ( 'appid' => '1' ...

  3. LOJ #6281. 数列分块入门 5-分块(区间开方、区间求和)

    #6281. 数列分块入门 5 内存限制:256 MiB时间限制:500 ms标准输入输出 题目类型:传统评测方式:文本比较 上传者: hzwer 提交提交记录统计测试数据讨论 5   题目描述 给出 ...

  4. jdk的split 有多坑

    先看段 代码: String str = "4117|519951|长信利泰灵活配置混合型证券投资基金|长信利泰|3|3||||156|0||||||||||||||||||||{\&quo ...

  5. (3) go 指针

    1.获取地址 取地址 &i 2.指针 var p *int=&i ptr 是一个地址 *ptr 是可看成一个变量,该地址所在的变量,也就是 num 3.常见值类型 引用类型 值类型:栈 ...

  6. 洛谷P1129 [ZJOI2007] 矩阵游戏

    题目传送门 分析:看到这题呢,首先想到的就是搜索,数据范围也不大嘛.但是仔细思考发现这题用搜索很难做,看了大佬们的题解后学到了,这一类题目要用二分图匹配来做.可以知道,如果想要的话,每一个子都可以移动 ...

  7. 【BZOJ 1119】 1119: [POI2009]SLO (置换)

    1119: [POI2009]SLO Description 对于一个1-N的排列(ai),每次你可以交换两个数ax与ay(x<>y),代价为W(ax)+W(ay) 若干次交换的代价为每次 ...

  8. 【Splay】bzoj3223 Tyvj 1729 文艺平衡树

    #include<cstdio> #include<iostream> #include<cstring> #include<algorithm> #i ...

  9. 猫、路由器、交换机和PC

    转载:http://duanzw102.blog.163.com/blog/static/161838173201392431722650/ 猫是 modem,是有网络供应商,比如电信公司提供的拨号工 ...

  10. String.format("%0"+length+"d", arr)中的%0和"d"分别代表什么

    public static void main(String[] args) { int a = 8; String s = String.format("%04d", a); S ...