使用nodejs写一个爬虫程序获取行政区划信息

2024-09-05 13:18

本文主要是介绍使用nodejs写一个爬虫程序获取行政区划信息,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近在开发一个房产管理系统,有个功能是需要对行政区划进行存入数据库管理,在网上找了很久,关于国家的行政区划数据都比较久远或者不完整,不能匹配自己的需求。
后来就想到用爬虫获取国家统计局网站的数据,因为Python还在学习中,所以就用自己熟悉的nodejs写了份简单的爬虫,能获取到国家统计局网站的全国行政区划信息(不包含港澳台)。

目录

  • 创建src文件价,作为生成的行政区划信息存放目录
  • 创建index.js 文件作为主要文件

初始化

使用npm初始化项目,按照提示输入相应内容(项目名称、版本、描述等信息)

npm init

安装依赖

  • cheerio: jquery核心功能的一个快速灵活而又简洁的实现,主要是为了用在服务器端需要对DOM进行操作的地方
  • axios:是一个基于 promise 的 HTTP 库(也可以使用nodejs的request或其他http库)
  • iconv-lite:解决nodejs中编码问题
  • async:区别与ES的async/await,这里作为一个库引入,主要使用async.mapLimit控制请求并发数

下文具体介绍各个库的使用

开发

分析目标网站

进入国家统计局网站,找到行政区划网页,目前最新的数据是2020年的,网址:http://www.stats.gov.cn/tjsj/tjbz/tjyqhdmhcxhfdm/2020/index.html

通过查看网页源码发现,其实这个网站比较简单,我们想获取的信息也直接在html内,所以可以直接使用cheerio来获取元素信息,并最终得到想要的数据。

对被爬取网站的分析应该放在第一步,然后再根据不同网站的分析结果,确定使用的依赖库。

首先定义几个变量

const HOST = 'http://www.stats.gov.cn/tjsj/tjbz/tjyqhdmhcxhfdm/2020/';
const headers = { 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/72.0.3626.119 Safari/537.36' };

HOST: 就是要爬取的网站,这里给单独出来,方便后面的操作,(可以进一步优化,把年份拿出来,每次获取时手动输入,获取指定年份的数据)
headers: 就是访问该网站时的header信息,这里就是为了模拟浏览器环境,防止出问题,其实不写也可以请求网站信息

  • 引入axios库,获取网站数据
    文件头部引入axios
const axios = require('axios');

然后定义一个函数用于请求地址获取相应数据。因为根据分析网站的结果可知,原网站数据是按层级一层一层划分并每个链接显示一级的数据。如:首页显示了所有省级信息,然后每个省级携带对应的不同链接,连接到下属城市,再根据城市信息链接到区县再到街道、居委会等。所以把请求接口的方法封装到一个函数里,通过参数访问,获取信息。

const fetchData = async (url) => {const res = await axios(url, { headers });

定义一个main函数,作为程序的主入口,再在main内执行

const main = async () => {const provinceData = await fetchData(HOST);
}main();

保存后执行node index.js即可获取网页信息。可用console打印出查看。

此时会发现,获取的数据,中文会乱码,这是因为,该网站编码采用的是GB2312。我们需要对获取的数据进行转译。

  • 引入iconv-lite
const iconv = require('iconv-lite');

修改请求函数:

let data = null;
const res = await axios(url, { responseType: 'arraybuffer', headers });
// iconv-lite 解析buffer数据为gb2312
data = iconv.decode(res.data, 'gb2312');

注意

axios会转换数据格式为utf-8,所以这里需要把获取的数据转换为流,再使用iconv-lite转为GB2312格式

获取到正确的数据后就可以使用cheerio进行精准的获取了。

  • 引入cheerio
const cheerio = require('cheerio');

分析网页结构,省级信息在table标签内,分四行显示,共四个tr标签,每个tr标签有个class="provincetr",然后名称在td下的a标签内,如下:

<tr class="provincetr"><td><a href="11.html">北京市<br></a></td><td><a href="12.html">天津市<br></a></td><td><a href="13.html">河北省<br></a></td><td><a href="14.html">山西省<br></a></td><td><a href="15.html">内蒙古自治区<br></a></td><td><a href="21.html">辽宁省<br></a></td><td><a href="22.html">吉林省<br></a></td><td><a href="23.html">黑龙江省<br></a></td>
</tr>

发现规律遍可以编写代码:

const proviceStr = (html) => {const $ = cheerio.load(html)let result = [];$(".provincetr a").each(function (index, element) {let name = $(element).text().trim();let url = $(element).attr("href");let id = url.replace('.html', '');result.push({pid: '',id,name,url: HOST+url,})});return result;
};

这里遍历class下的元素,获取a标签内的内容和href信息,并需要获取对应的省级id(使用了通过链接截取,其实href内的信息也是对应的id)。并把url信息保存下来,用于下一级数据获取的一句

打印result应该就能看到获取的数据了。

接下来需要把数据存到文件中

  • 引入nodejs的fs模块
const fs = require('fs');

定义存放的变量,为了方便,直接把各个级别的信息分开存储:

const filePath = {province: 'src/province.json',city: 'src/city.json',country: 'src/country.json'
}

修改main方法:

const main = async () => {const Index = joinUrl('index.html');const provinceData = await fetchData(Index, 'province', '');fs.writeFileSync(filePath.province, JSON.stringify(provinceData));
}

此时打开src文件夹便能看见多了个province.json,打开是个压缩的json文件。

然后再分析市级数据。市级网页结构布局与省级类似,只是变成纵向排列,tr标签classcitytr。区县和街道、居委会布局都一样,只是class不同,所以可以抽取出共同代码,封装成一个函数。

需要注意的是,越往下级爬取,数据量会越来越大,这时可能会报错,甚至会被原网站封ip的风险。

这里因为只是爬取到区县级数据,所以没有做过多考虑,只是引入的async库的mapLimit,用于并发请求。

  • 最后,配置package.json,使用npm脚本启动项目,不再使用node index.js

时间仓储,只是为了满足项目中的一个小需求,临时写的一个爬虫,也是第一次使用nodejs爬取网页,问题还有很多,希望多多指正。后续抽出时间,对代码进行完善,解决爬取数据量大就报错的问题,并对省级id进行位数完善等。

  • 项目地址
    https://github.com/imchaoyu/node-get-districts

这篇关于使用nodejs写一个爬虫程序获取行政区划信息的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1139068

相关文章

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

C#使用Spire.Doc for .NET实现HTML转Word的高效方案

《C#使用Spire.Docfor.NET实现HTML转Word的高效方案》在Web开发中,HTML内容的生成与处理是高频需求,然而,当用户需要将HTML页面或动态生成的HTML字符串转换为Wor... 目录引言一、html转Word的典型场景与挑战二、用 Spire.Doc 实现 HTML 转 Word1

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

MyBatis ParameterHandler的具体使用

《MyBatisParameterHandler的具体使用》本文主要介绍了MyBatisParameterHandler的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录一、概述二、源码1 关键属性2.setParameters3.TypeHandler1.TypeHa

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同

使用Python实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方