Node.js+jade+mongodb+mongoose實(shí)現(xiàn)爬蟲分離入庫(kù)與生成靜態(tài)文件的方法
接著這篇文章Node.js+jade抓取博客所有文章生成靜態(tài)html文件的實(shí)例繼續(xù),在這篇文章中實(shí)現(xiàn)了采集與靜態(tài)文件的生成,在實(shí)際的采集項(xiàng)目中, 應(yīng)該是先入庫(kù)再選擇性的生成靜態(tài)文件。
那么我選擇的數(shù)據(jù)庫(kù)是mongodb,為什么用這個(gè)數(shù)據(jù)庫(kù),因?yàn)檫@個(gè)數(shù)據(jù)庫(kù)是基于集合,數(shù)據(jù)的操作基本是json,與dom模塊cheerio具有非常大的親和力,cheerio處理過(guò)濾出來(lái)的數(shù)據(jù),可以直接插入mongodb,不需要經(jīng)過(guò)任何的處理,非常的便捷,當(dāng)然跟node.js的親和力那就不用說(shuō)了,更重要的是,性能很棒。這篇文章我就不具體寫mongodb的基本用法,到時(shí)候會(huì)另起文章從0開始寫mongodb基本常用用法.先看下入庫(kù)的效果與生成靜態(tài)文件的效果:


我在這個(gè)階段,把爬蟲分離成2個(gè)模塊,采集入庫(kù)( crawler.js ), 生成靜態(tài)文件(makeHtml.js).
crawler.js:
var http = require('http');
var cheerio = require('cheerio');
var mongoose = require('mongoose');
mongoose.Promise = global.Promise;
var DB_URL = 'mongodb://localhost:27017/crawler';
var aList = []; //博客文章列表信息
var aUrl = []; //博客所有的文章url
var db = mongoose.createConnection(DB_URL);
db.on('connected', function (err) {
if (err) {
console.log(err);
} else {
console.log('db connected success');
}
});
var Schema = mongoose.Schema;
var arcSchema = new Schema({
id: Number, //文章id
title: String, //文章標(biāo)題
url: String, //文章鏈接
body: String, //文章內(nèi)容
entry: String, //摘要
listTime: Date //發(fā)布時(shí)間
});
var Article = db.model('Article', arcSchema);
function saveArticle(arcInfo) {
var arcModel = new Article(arcInfo);
arcModel.save(function (err, result) {
if (err) {
console.log(err);
} else {
console.log(`${arcInfo['title']} 插入成功`);
}
});
}
function filterArticle(html) {
var $ = cheerio.load(html);
var arcDetail = {};
var title = $("#cb_post_title_url").text();
var href = $("#cb_post_title_url").attr("href");
var re = /\/(\d+)\.html/;
var id = href.match(re)[1];
var body = $("#cnblogs_post_body").html();
return {
id: id,
title: title,
url: href,
body: body
};
}
function crawlerArc(url) {
var html = '';
var str = '';
var arcDetail = {};
http.get(url, function (res) {
res.on('data', function (chunk) {
html += chunk;
});
res.on('end', function () {
arcDetail = filterArticle(html);
saveArticle(arcDetail);
if ( aUrl.length ) {
setTimeout(function () {
if (aUrl.length) {
crawlerArc(aUrl.shift());
}
}, 100);
}else {
console.log( '采集任務(wù)完成' );
return;
}
});
});
}
function filterHtml(html) {
var $ = cheerio.load(html);
var arcList = [];
var aPost = $("#content").find(".post-list-item");
aPost.each(function () {
var ele = $(this);
var title = ele.find("h2 a").text();
var url = ele.find("h2 a").attr("href");
ele.find(".c_b_p_desc a").remove();
var entry = ele.find(".c_b_p_desc").text();
ele.find("small a").remove();
var listTime = ele.find("small").text();
var re = /\d{4}-\d{2}-\d{2}\s*\d{2}[:]\d{2}/;
listTime = listTime.match(re)[0];
arcList.push({
title: title,
url: url,
entry: entry,
listTime: listTime
});
});
return arcList;
}
function nextPage(html) {
var $ = cheerio.load(html);
var nextUrl = $("#pager a:last-child").attr('href');
if (!nextUrl) return getArcUrl(aList);
var curPage = $("#pager .current").text();
if (!curPage) curPage = 1;
var nextPage = nextUrl.substring(nextUrl.indexOf('=') + 1);
if (curPage < nextPage) crawler(nextUrl);
}
function crawler(url) {
http.get(url, function (res) {
var html = '';
res.on('data', function (chunk) {
html += chunk;
});
res.on('end', function () {
aList.push(filterHtml(html));
nextPage(html);
});
});
}
function getArcUrl(arcList) {
for (var key in arcList) {
for (var k in arcList[key]) {
aUrl.push(arcList[key][k]['url']);
}
}
crawlerArc(aUrl.shift());
}
var url = 'http://www.cnblogs.com/ghostwu/';
crawler(url);
其他的核心模塊沒(méi)有怎么改動(dòng),主要增加了數(shù)據(jù)庫(kù)連接,數(shù)據(jù)庫(kù)創(chuàng)建,集合創(chuàng)建( 集合相當(dāng)于關(guān)系型數(shù)據(jù)庫(kù)中的表 ),Schema( 相當(dāng)于關(guān)系型數(shù)據(jù)庫(kù)的表結(jié)構(gòu) ).
mongoose操作數(shù)據(jù)庫(kù)( save:插入數(shù)據(jù) ).分離了文件生成模塊.
makeHtml.js文件
var fs = require('fs');
var jade = require('jade');
var mongoose = require('mongoose');
mongoose.Promise = global.Promise;
var DB_URL = 'mongodb://localhost:27017/crawler';
var allArc = [];
var count = 0;
var db = mongoose.createConnection(DB_URL);
db.on('connected', function (err) {
if (err) {
console.log(err);
} else {
console.log('db connected success');
}
});
var Schema = mongoose.Schema;
var arcSchema = new Schema({
id: Number, //文章id
title: String, //文章標(biāo)題
url: String, //文章鏈接
body: String, //文章內(nèi)容
entry: String, //摘要
listTime: Date //發(fā)布時(shí)間
});
var Article = db.model('Article', arcSchema);
function makeHtml(arcDetail) {
str = jade.renderFile('./views/layout.jade', arcDetail);
++count;
fs.writeFile('./html/' + count + '.html', str, function (err) {
if (err) {
console.log(err);
}
console.log( `${arcDetail['id']}.html創(chuàng)建成功` + count );
if ( allArc.length ){
setTimeout( function(){
makeHtml( allArc.shift() );
}, 100 );
}
});
}
function getAllArc(){
Article.find( {}, function( err, arcs ){
allArc = arcs;
makeHtml( allArc.shift() );
} ).sort( { 'id' : 1 } );
}
getAllArc();
以上這篇Node.js+jade+mongodb+mongoose實(shí)現(xiàn)爬蟲分離入庫(kù)與生成靜態(tài)文件的方法就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- node.js連接mongoose數(shù)據(jù)庫(kù)方法詳解
- node.js使用mongoose操作數(shù)據(jù)庫(kù)實(shí)現(xiàn)購(gòu)物車的增、刪、改、查功能示例
- node.js利用mongoose獲取mongodb數(shù)據(jù)的格式化問(wèn)題詳解
- Node.js中使用mongoose操作mongodb數(shù)據(jù)庫(kù)的方法
- Node.js的MongoDB驅(qū)動(dòng)Mongoose基本使用教程
- 安裝使用Mongoose配合Node.js操作MongoDB的基礎(chǔ)教程
- node.js mongoose index索引操作
相關(guān)文章
nodejs利用http模塊實(shí)現(xiàn)銀行卡所屬銀行查詢和騷擾電話驗(yàn)證示例
本篇文章主要介紹了nodejs利用http模塊實(shí)現(xiàn)銀行卡所屬銀行查詢和騷擾電話驗(yàn)證示例,有興趣的可以了解一下。2016-12-12
node.js中的fs.lchownSync方法使用說(shuō)明
這篇文章主要介紹了node.js中的fs.lchownSync方法使用說(shuō)明,本文介紹了fs.lchownSync的方法說(shuō)明、語(yǔ)法、接收參數(shù)、使用實(shí)例和實(shí)現(xiàn)源碼,需要的朋友可以參考下2014-12-12
在CentOS 7服務(wù)器上安裝Node.js的方法步驟
Node.js 是一個(gè)用于服務(wù)器端編程的 JavaScript 平臺(tái),允許用戶快速構(gòu)建網(wǎng)絡(luò)應(yīng)用程序,通過(guò)在前端和后端都使用 JavaScript,開發(fā)可以更加一致并且可以在同一個(gè)系統(tǒng)中設(shè)計(jì),在本指南中,我們將向您展示如何在 Ubuntu 14.04 服務(wù)器上開始使用 Node.js2024-09-09
no-vnc和node.js實(shí)現(xiàn)web遠(yuǎn)程桌面的完整步驟
這篇文章主要給大家介紹了關(guān)于no-vnc和node.js實(shí)現(xiàn)web遠(yuǎn)程桌面的完整步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
NodeJS Web應(yīng)用監(jiān)聽sock文件實(shí)例
這篇文章主要介紹了NodeJS Web應(yīng)用監(jiān)聽sock文件實(shí)例,本文講解 NodeJS 的 TCP 和 HTTP 監(jiān)聽 Domain Socket 文件例子,需要的朋友可以參考下2015-02-02
node+koa2+mysql+bootstrap搭建一個(gè)前端論壇
本篇文章通過(guò)實(shí)例給大家分享了用node+koa2+mysql+bootstrap搭建一個(gè)前端論壇的步驟,有需要的朋友參考下。2018-05-05
NodeJs生成sitemap站點(diǎn)地圖的方法示例
這篇文章主要介紹了NodeJs生成sitemap站點(diǎn)地圖的方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06

