详解nodejs爬虫程序解决gbk等中文编码问题
发布时间 - 2026-01-11 00:32:12 点击率:次使用nodejs写了一个爬虫的demo,目的是提取网页的title部分。

遇到最大的问题就是网页的编码与nodejs默认编码不一致造成的乱码问题。nodejs支持utf8, ucs2, ascii, binary, base64, hex等编码方式,但是对于汉语言来说编码主要分为三种,utf-8,gb2312,gbk。这里面gbk是完全兼容gb2312的,因此在处理编码的时候主要就分为utf-8以及gbk两大类。(这是在没有考虑到其他国家的编码情况,比如日本的Shift_JIS编码等,同时这里这个iconv-lite模块支持的编码方法有限)。
首先说一下浏览器显示网页内容的时候是如何处理编码问题的。服务器和客户端进行通信,服务端将网页按照指定的编码方式(比如gbk)编码成为二进制码流(即我们使用wireshark抓包看到额16进制码流)传送给我们的客户端。客户端则会根据网页源码中所规定的编码方式,由浏览器调用对应的解码器,将二进制码流解码后显示出来。而编码方式通常在网页中是如下内容表示:
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
或者
<meta charset=utf-8"/>
如果客户端是nodejs爬虫请求程序,由于nodejs默认的编码方式是utf-8,因此爬虫程序将接收到的二进制码流以字符串(默认方式utf-8)显示的时候则会显示乱码。这个时候需要将原始的二进制码流按照网页原来的编码方式解码,则不会出现乱码。
因此解决方法如下:
将接收到的网页源码以二进制的方式存储下来,处理二进制数据流使用Buffer全局对象。
res.on('data', function(data) {
htmlData.push(data);
htmlDataLength += data.length;
});
var bufferHtmlData = Buffer.concat(htmlData,htmlDataLength);
然后对这些二进制的数据调用对应的解码程序。iconv-lite模块用于解码,cheerio模块用于解析网页内容。
decodeHtmlData = iconv.decode(bufferHtmlData,'gbk');
var $ = cheerio.load(decodeHtmlData, {decodeEntities: false});
$('title','head').each(function(i, e) {
htmlHeadTitle = $(e).text();
console.log(htmlHeadTitle);
});
上述bufferHtmlData为二进制码流,decodeHtmlData为将二进制码流通过gbk编码规则转换为unicode编码对应的数字(即usc2字节流),然后在转换为对应的字符串。下述为iconv-lite源码中解码部分,地址在这里:
fromEncoding: function(buf) {
buf = ensureBuffer(buf);
var idx = 0, len = 0,
newBuf = new Buffer(len*2),unicode,gbkcode;
for (var i = 0, _len = buf.length; i < _len; i++, len++) {
if (!!(buf[i] & 0x80)) {//the high bit is 1, so this byte is gbkcode's high byte.skip next byte
i++;
}
}
var newBuf = new Buffer(len*2);
for (var i = 0, j = 0, _len = buf.length; i < _len; i++, j++) {
var temp = buf[i], gbkcode, unicode;
if (temp & 0x80) {
gbkcode = (temp << 8) + buf[++i];
unicode = table[gbkcode] || iconv.defaultCharUnicode.charCodeAt(0);//not found in table, replace with defaultCharUnicode
}else {
unicode = temp;
}
newBuf[j*2] = unicode & 0xFF;//low byte
newBuf[j*2+1] = unicode >> 8;//high byte
}
return newBuf.toString('ucs2');
}
可以看到最终返回的是newBuf.toString(‘ucs2')字符串。
爬虫程序源码如下:
var cheerio = require('cheerio');
var http = require('http');
var iconv = require('iconv-lite');
var htmlData = [];
var htmlDataLength = 0;
var count = 0;
http.globalAgent = 'Mozilla/5.0 (Windows NT 6.1; rv:2.0.1) Gecko/20100101 Firefox/4.0.1';
http.get('http://www.cr173.com', function(res) {
res.on('data', function(data) {
htmlData.push(data);
htmlDataLength += data.length;
count ++;
});
res.on('end',function(){
callback(htmlData);
});
});
function callback(htmlData){
console.log(count);
var bufferHtmlData = Buffer.concat(htmlData,htmlDataLength);
var charset = '';
var decodeHtmlData;
var htmlHeadTitle = '';
var htmlHeadCharset = '';
var htmlHeadContent = '';
var index = 0;
var $ = cheerio.load(bufferHtmlData, {decodeEntities: false});
$('meta','head').each(function(i, e) {
htmlHeadCharset = $(e).attr('charset');
htmlHeadContent = $(e).attr('content');
if(typeof(htmlHeadCharset) != 'undefined'){
charset = htmlHeadCharset;
}
if(typeof(htmlHeadContent) != 'undefined'){
if(htmlHeadContent.match(/charset=/ig)){
index = htmlHeadContent.indexOf('=');
charset = htmlHeadContent.substring(index+1);
}
}
});
//此处为什么需要对整个网页进行转吗,是因为cheerio这个组件不能够返回buffer,iconv则无法转换之
if(charset.match(/gb/ig)){
decodeHtmlData = iconv.decode(bufferHtmlData,'gbk');
}
else{//因为有可能返回的网页中不存在charset字段,因此默认都是按照utf8进行处理
decodeHtmlData = iconv.decode(bufferHtmlData,'utf8');
}
var $ = cheerio.load(decodeHtmlData, {decodeEntities: false});
$('title','head').each(function(i, e) {
htmlHeadTitle = $(e).text();
console.log(htmlHeadTitle);
});
console.log(charset);
}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# nodejs
# gbk编码
# gbk
# 中文编码
# NodeJS制作爬虫全过程
# 详解Node使用Puppeteer完成一次复杂的爬虫
# 简单好用的nodejs 爬虫框架分享
# Node.js编写爬虫的基本思路及抓取百度图片的实例分享
# 基于Node.js的强大爬虫 能直接发布抓取的文章哦
# nodejs爬虫抓取数据乱码问题总结
# NodeJS制作爬虫全过程(续)
# nodeJs爬虫获取数据简单实现代码
# node+express制作爬虫教程
# Node.js爬虫如何获取天气和每日问候详解
# 客户端
# 转换为
# 则会
# 的是
# 都是
# 这是
# 在这里
# 是因为
# 有可能
# 给我们
# 可以看到
# 三种
# 考虑到
# 写了
# 这个时候
# 日本
# 这里面
# 不能够
# 中不
# 如何处理
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
laravel怎么配置和使用PHP-FPM来优化性能_laravel PHP-FPM配置与性能优化方法
Android仿QQ列表左滑删除操作
Laravel如何处理和验证JSON类型的数据库字段
JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)
Laravel广播系统如何实现实时通信_Laravel Reverb与WebSockets实战教程
Laravel怎么集成Log日志记录_Laravel单文件与每日日志配置及自定义通道【详解】
Laravel怎么返回JSON格式数据_Laravel API资源Response响应格式化【技巧】
HTML透明颜色代码在Angular里怎么设置_Angular透明颜色使用指南【详解】
ChatGPT回答中断怎么办 引导AI继续输出完整内容的方法
在线制作视频网站免费,都有哪些好的动漫网站?
详解jQuery中的事件
Android GridView 滑动条设置一直显示状态(推荐)
Laravel中的withCount方法怎么高效统计关联模型数量
Laravel定时任务怎么设置_Laravel Crontab调度器配置
如何快速生成凡客建站的专业级图册?
Laravel如何集成微信支付SDK_Laravel使用yansongda-pay实现扫码支付【实战】
Win11怎样安装网易有道词典_Win11安装词典教程【步骤】
Laravel如何创建自定义中间件?(Middleware代码示例)
网站制作软件免费下载安装,有哪些免费下载的软件网站?
惠州网站建设制作推广,惠州市华视达文化传媒有限公司怎么样?
如何用美橙互联一键搭建多站合一网站?
深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?
js代码实现下拉菜单【推荐】
微信公众帐号开发教程之图文消息全攻略
如何用PHP工具快速搭建高效网站?
如何用PHP快速搭建高效网站?分步指南
Laravel Octane如何提升性能_使用Laravel Octane加速你的应用
活动邀请函制作网站有哪些,活动邀请函文案?
制作公司内部网站有哪些,内网如何建网站?
制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?
如何破解联通资金短缺导致的基站建设难题?
桂林网站制作公司有哪些,桂林马拉松怎么报名?
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
Laravel如何使用API Resources格式化JSON响应_Laravel数据资源封装与格式化输出
Windows10电脑怎么查看硬盘通电时间_Win10使用工具检测磁盘健康
如何快速搭建高效香港服务器网站?
Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)
浅谈redis在项目中的应用
百度输入法ai组件怎么删除 百度输入法ai组件移除工具
Swift开发中switch语句值绑定模式
Laravel storage目录权限问题_Laravel文件写入权限设置
Laravel如何记录日志_Laravel Logging系统配置与自定义日志通道
西安专业网站制作公司有哪些,陕西省建行官方网站?
Python3.6正式版新特性预览
zabbix利用python脚本发送报警邮件的方法
大同网页,大同瑞慈医院官网?
高性能网站服务器部署指南:稳定运行与安全配置优化方案
html5如何设置样式_HTML5样式设置方法与CSS应用技巧【教程】
使用PHP下载CSS文件中的所有图片【几行代码即可实现】

