使用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文件的顶部声明编码通常是必须的,尤其是在处理非ASCII字符时,下面就来介绍一下两种头部文件声明,具有一定的参考价值,感兴趣的可以了解一下... 目录一、# coding=utf-8二、#!/usr/bin/env python三、运行Python程序四、

MySQL 事务的概念及ACID属性和使用详解

《MySQL事务的概念及ACID属性和使用详解》MySQL通过多线程实现存储工作,因此在并发访问场景中,事务确保了数据操作的一致性和可靠性,下面通过本文给大家介绍MySQL事务的概念及ACID属性和... 目录一、什么是事务二、事务的属性及使用2.1 事务的 ACID 属性2.2 为什么存在事务2.3 事务

Linux查看系统盘和SSD盘的容量、型号及挂载信息的方法

《Linux查看系统盘和SSD盘的容量、型号及挂载信息的方法》在Linux系统中,管理磁盘设备和分区是日常运维工作的重要部分,而lsblk命令是一个强大的工具,它用于列出系统中的块设备(blockde... 目录1. 查看所有磁盘的物理信息方法 1:使用 lsblk(推荐)方法 2:使用 fdisk -l(

使用Python实现网页表格转换为markdown

《使用Python实现网页表格转换为markdown》在日常工作中,我们经常需要从网页上复制表格数据,并将其转换成Markdown格式,本文将使用Python编写一个网页表格转Markdown工具,需... 在日常工作中,我们经常需要从网页上复制表格数据,并将其转换成Markdown格式,以便在文档、邮件或

Python使用pynput模拟实现键盘自动输入工具

《Python使用pynput模拟实现键盘自动输入工具》在日常办公和软件开发中,我们经常需要处理大量重复的文本输入工作,所以本文就来和大家介绍一款使用Python的PyQt5库结合pynput键盘控制... 目录概述:当自动化遇上可视化功能全景图核心功能矩阵技术栈深度效果展示使用教程四步操作指南核心代码解析

使用Python获取JS加载的数据的多种实现方法

《使用Python获取JS加载的数据的多种实现方法》在当今的互联网时代,网页数据的动态加载已经成为一种常见的技术手段,许多现代网站通过JavaScript(JS)动态加载内容,这使得传统的静态网页爬取... 目录引言一、动态 网页与js加载数据的原理二、python爬取JS加载数据的方法(一)分析网络请求1

SpringCloud使用Nacos 配置中心实现配置自动刷新功能使用

《SpringCloud使用Nacos配置中心实现配置自动刷新功能使用》SpringCloud项目中使用Nacos作为配置中心可以方便开发及运维人员随时查看配置信息,及配置共享,并且Nacos支持配... 目录前言一、Nacos中集中配置方式?二、使用步骤1.使用$Value 注解2.使用@Configur

Mac备忘录怎么导出/备份和云同步? Mac备忘录使用技巧

《Mac备忘录怎么导出/备份和云同步?Mac备忘录使用技巧》备忘录作为iOS里简单而又不可或缺的一个系统应用,上手容易,可以满足我们日常生活中各种记录的需求,今天我们就来看看Mac备忘录的导出、... 「备忘录」是 MAC 上的一款常用应用,它可以帮助我们捕捉灵感、记录待办事项或保存重要信息。为了便于在不同

通过cmd获取网卡速率的代码

《通过cmd获取网卡速率的代码》今天从群里看到通过bat获取网卡速率两段代码,感觉还不错,学习bat的朋友可以参考一下... 1、本机有线网卡支持的最高速度:%v%@echo off & setlocal enabledelayedexpansionecho 代码开始echo 65001编码获取: >

如何Python使用设置word的页边距

《如何Python使用设置word的页边距》在编写或处理Word文档的过程中,页边距是一个不可忽视的排版要素,本文将介绍如何使用Python设置Word文档中各个节的页边距,需要的可以参考下... 目录操作步骤代码示例页边距单位说明应用场景与高级用China编程途小结在编写或处理Word文档的过程中,页边距是一个