!ta竟然用Node.js对这些beauty图做了这些事。。。

news/2024/7/1 21:12:49

做了什么

一个用于爬取www.nvshens.com上妹子图片的爬虫。如有侵权,马上关闭
爬虫

原因

一张张下实在太麻烦了
爬虫结果

如何使用

0. node -v >= 7.6
1. git clone https://github.com/laihaibo/beauty-spider.git
2. npm i
3. npm run start (爬取相册图片链接,并保存为json)
4. npm run calc (获取爬取的相册数和文件数)
5. npm run download (下载图片文件)

update

against反爬虫

图片下载完之后会发现变成了盗链图片。于是观察浏览器正常浏览行为。在请求头中设置referer, acceptuser-agent。解决该问题

request.get(url).set({
        'Referer': 'https://www.google.com',
        'Accept': 'image/webp,image/*,*/*;q=0.8',
        'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3091.0 Safari/537.36'
      }).end((err, res) => {})

断线继续下载

图片下载700个文件时,经常断线。应该是网站的饭爬虫机制起了作用,暂时无法解决。重新下载时理应跳过已经下载的文件。于是在保存图片时会先判断图片是否存在。

let isExit = fs.existsSync(path);
if (!isExit) {
  saveOne(...args)
}

获取理应下载的相册数和文件数

let data = JSON.parse(fs.readFileSync(path));

let count = data.reduce((prev, cur) => prev + cur.imgList.length, 0);

console.log(`共${data.length}个相册,共${count}张图片`);

步骤

  1. 引入所需的库

    const fs = require("fs");
    const mkdirp = require('mkdirp');
    const cheerio = require('cheerio');
    const request = require('superagent');
    require('superagent-charset')(request);
  2. 页面分析,配置config文件
    分析相册地址,以韩国这个标签为例,首页为https://www.nvshens.com/gallery/hanguo/, 第二页为https://www.nvshens.com/gallery/hanguo/2.html

    const config = {
      current: 'hanguo',
      allTags: {
        rougan: `https://www.nvshens.com/gallery/rougan/`,
        hanguo: 'https://www.nvshens.com/gallery/hanguo/'
      }
    }
  3. 封装获取指定url的html内容函数

    //该网站编码为utf-8
    const getHtml = url => {
      return new Promise((resolve, reject) => {
        request.get(url).charset('utf-8').end((err, res) => {
          err ? reject(err) : resolve(cheerio.load(res.text));
        })
      })
    }
  4. 获取本分类下所有相册的标签

    /**
     * @param {string} startUrl 标签首页的url地址
     */
    const getAlbums = (startUrl) => {
      return new Promise((resolve, reject) => {
        let albums = [];  // 用于保存该标签的所有相册信息
    
        let getQuery = async startUrl => {
          try {
            let $ = await getHtml(startUrl);
            let pages = $('#listdiv .pagesYY a').length;  // 获取页数
    
            for (let i = 1; i <= pages; i++) {
              let pageUrl = `${startUrl + i}.html`  // 设置每页的url
              let $ = await getHtml(pageUrl);
    
              // 动态设置pages的值
              let compare = $('#listdiv .pagesYY a').map(function (i, el) {
                return parseInt($(this).text(), 0);
              }).get().filter(x => x > 0);
              pages = conmpare.length < 2 ? pages : compare.reduce((prev, cur) => Math.max(prev, cur));
    
              $('.galleryli_title a').each(function () {
                albums.push({
                  title: $(this).text(),
                  url: `https://www.nvshens.com${$(this).attr("href")}`,
                  imgList: [],
                  id: parseInt($(this).attr("href").split('/')[2], 10)
                })
              })
            }
    
            resolve(albums);  // 返回相册信息
          } catch (error) {
            console.log(error);
          }
        }
    
        getQuery(startUrl);
      })
    }
  5. 获取所有相册的图片信息

    /**
     * @param {string} startUrl 该相册首页的url地址
     */
    const getImgList = (startUrl) => {
      return new Promise((resolve, reject) => {
        let albums = [];  // 存储本相册的所有图片信息
    
        let getQuery = async startUrl => {
          try {
            let $ = await getHtml(startUrl);
            let pages = $('#pages a').length;
    
            for (let i = 1; i <= pages; i++) {
              let pageUrl = `${startUrl + i}.html`
              let $ = await getHtml(pageUrl);
    
              $('#hgallery img').each(function () {
    
                let url = $(this).attr('src');  //图片地址
                let fileName = url.split('/').pop();  //文件名
                let id = parseInt(fileName.split('.')[0], 10); //id
    
                albums.push({
                  url,
                  fileName,
                  id
                })
              })
            }
    
            resolve(albums); // 返回本相册的所有图片信息
          } catch (error) {
            console.log(error);
          }
        }
    
        getQuery(startUrl);
      })
    }
  6. 保存相册信息

    /**
     * @param {string} path 保存数据的路径
     * @param {array} albums 相册信息数组
     */
    const saveData = (path, albums) => {
        fs.writeFile(path, JSON.stringify(albums, null, ' '), function (err) {
            err ? console.log(err) : console.log('Data saved');
        });
    }
  7. 保存图片

    /**
     12. @param {string} title 图片所在文件夹名
     13. @param {string} url 图片url
     14. @param {string} fileName 图片名
     15. @param {array} imgList 单个相册的图片信息
     */
    // 保存一张图片
    const saveOne = (title, url, fileName) => {
      return new Promise((resolve, reject) => {
        let path = `./img/${currentImgType}/${title}/${fileName}`;
        request.get(url).end((err, res) => {
          if (err) {
            console.log(`Error: ${err} in getting ${url}`)
          }
          fs.writeFile(path, res.body, function (err) {
            if (err) console.log(`Error: ${err} in downloading ${url}`)
          });
          resolve();
        })
      })
    }
    
    //保存一个相册下的多张图片
    const saveImg = ({title,imgList}) => {
      // 创建文件夹
      mkdirp(`./img/${currentImgType}/${title}`, function (err) {
        if (err) {
          console.log(`Error: ${err} in makedir ${title}`);
        }
      });
    
      let getQuery = async() => {
        try {
          for (let {url,fileName} of imgList) {
            await saveOne(title, url, fileName);
          }
        } catch (error) {
          console.log(error);
        }
      }
    
      // 打印下载一个相册所需时间
      console.time(`download ${title}...`)
      getQuery();
      console.timeEnd(`download ${title}...`)
    }
  8. 执行爬虫

    const doSpider = async() => {
      try {
        // 获取相册信息
        let albums = await getAlbums(allTags[current]);
    
        // 获取每张图片信息
        for (let album of albums) {
          let imgList = await getImgList(album.url);
          album.imgList = imgList;
        }
    
        // 保存json
        let jsonPath = `./data`;
        mkdirp(jsonPath, function (err) {
          if (err) {
            console.log(`Error: ${err} in makedir of Json`);
          }
        });
        saveData(`${jsonPath}/${currentImgType}.json`, albums);
    
        // 保存图片
        for (let value of albums) {
          saveImg(value)
        }
    
      } catch (error) {
        console.log(error);
      }
    }

心得体会

  1. 有些坑如果不踩过一遍是不会吐血的,比如cheerio的操作和fs的操作

  2. just do it

感谢

本文有参考nieheyong的HanhandeSpider和其他的爬虫文章,得到很多启发


http://www.niftyadmin.cn/n/4054453.html

相关文章

ceph(4)--Ceph 的基础数据结构

本系列文章会深入研究 Ceph 以及 Ceph 和 OpenStack 的集成&#xff1a; &#xff08;1&#xff09;安装和部署 &#xff08;2&#xff09;Ceph RBD 接口和工具 &#xff08;3&#xff09;Ceph 物理和逻辑结构 &#xff08;4&#xff09;Ceph 的基础数据结构 &#xff08;5&…

使用jwt记录

pom依赖 <!-- jwt依赖--><dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt</artifactId><version>0.9.1</version></dependency> 代码&#xff1a; package com.example.demo;import io.jsonw…

记录SpringBoot和SpringCloud版本对应关系

官方参考&#xff1a;https://spring.io/projects/spring-cloud SpringCloud版本SpringBoot版本Hoxton 2.2.x2.2.xGreenwich 2.1.x2.1.xFinchley 2.0.x2.0.xEdgware 1.5.x1.5.xDalston 1.5.x1.5.x

基于javax的websocket服务端实现心跳机制

websocket连接类 package com.dnn.controller.inter;import java.io.IOException; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; import java.util.Set; import java.util.concurrent.CopyOnWriteArraySet;import javax.websocket.…

js的clientXY、pageXY、screenXY、offsetXY属性对比

一、clientX、clientY 点击位置距离当前body可视区域的x&#xff0c;y坐标 二、pageX、pageY 对于整个页面来说&#xff0c;包括了被卷去的body部分的长度 三、screenX、screenY 点击位置距离当前电脑屏幕的x&#xff0c;y坐标 四、offsetX、offsetY 相对于带有定位的父盒子的x…

ElasticSearch使用kibana控制台查询示例(时间范围查询)

记录一下日期查询问题 "format": "yyyy-MM-dd HH:mm:ss||yyyy-MM-dd||epoch_millis"# 创建索引 PUT my_date1 {"mappings": {"properties": {"publicDate": {"type": "date","format": &quo…

SpringBoot整合Quartz 实现分布式定时任务调度

一、Quartz 集群架构 Quartz 是 Java 领域最著名的开源任务调度工具。 在上篇文章中&#xff0c;我们详细的介绍了 Quartz 的单体应用实践&#xff0c;如果只在单体环境中应用&#xff0c;Quartz 未必是最好的选择&#xff0c;例如Spring Scheduled一样也可以实现任务调度&am…

ElasticSearch6.X版本自动添加时间戳

需求&#xff1a;根据时间提取es数据 解决&#xff1a;为es的记录添加时间戳 1、方法 配置时间戳 pipeline PUT _ingest/pipeline/my_timestamp_pipeline {"description": "Adds a field to a document with the time of ingestion","processors&q…