成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

用Promise實現(xiàn)隊列(爬一爬慕課網(wǎng)HTML代碼)

XFLY / 3533人閱讀

摘要:項目初始化創(chuàng)建一個文件,快捷創(chuàng)建非常簡單?;臼褂梅椒ň唧w的請自行點擊連接查看喲頁面數(shù)據(jù)解析是一個的庫,可以理解為一個版本的,用來從網(wǎng)頁中以取數(shù)據(jù),使用方式和基本相同。執(zhí)行數(shù)組中每個值的函數(shù),包含四個參數(shù)作為第一次調(diào)用的第一個參數(shù)。

項目初始化

創(chuàng)建一個package.json文件,webstorm快捷創(chuàng)建package.json非常簡單。
使用 npm init 快速創(chuàng)建。

工具模塊

需要下載的的模塊

superagent 頁面數(shù)據(jù)下載

cheerio 頁面數(shù)據(jù)解析

這是2個npm包,我們先下載:

npm install superagent cheerio --save

需要引入的模塊

fs

path

引入項目依賴
const cheerio = require("cheerio");
const agent = require("superagent");
const path = require("path");
const fs = require("fs");
定義地址數(shù)組

我們希望以隊列的形式逐個對這些地址進行訪問,獲取HTML代碼,以便后續(xù)處理:

const urls = [{
    page:1,
    url:"https://www.imooc.com/course/list?c=fe&page=1"
},{
    page:2,
    url:"https://www.imooc.com/course/list?c=fe&page=2"
},{
    page:3,
    url:"https://www.imooc.com/course/list?c=fe&page=3"
}];
定義數(shù)據(jù)結(jié)構(gòu)

慕課網(wǎng)課程列表:

對此我們定義如下的數(shù)據(jù)結(jié)構(gòu):

 [
     {
      page: 1,
      data: [
          {
             title:"", // 課程標題
             imgurl:"", // 課程圖片
             level:"", // 等級
             studynum:0, // 學習人數(shù)
             description:"xxxx" // 課程描述
        }
        ...... // 每一個頁面有多條課程信息
      ]  
    }
    ...... // 一共有多個頁面
 ]
superagent 頁面數(shù)據(jù)下載
superagent是nodejs里一個非常方便的客戶端請求代碼模塊,superagent是一個輕量級的,漸進式的ajax API,可讀性好,學習曲線低,內(nèi)部依賴nodejs原生的請求API,適用于nodejs環(huán)境下。

基本使用方法:
具體的請自行點擊連接查看喲...

request
    .get("/login")
    .end(function(err, res){
        // code
    });
cheerio 頁面數(shù)據(jù)解析
cheerio是一個node的庫,可以理解為一個Node.js版本的jquery,用來從網(wǎng)頁中以 css selector取數(shù)據(jù),使用方式和jquery基本相同。

需要先loading一個需要加載html文檔,后面就可以jQuery一樣使用操作頁面了。

基本使用方法:
具體的請自行點擊連接查看喲...

const cheerio = require("cheerio");
const $ = cheerio.load("
    ...
"); $("#fruits").addClass("newClass");
使用Promise實現(xiàn)隊列

這才是本篇文章的重頭戲...

我們需要使用數(shù)組的一個方法 reduce()

arr.reduce([callback, initialValue])

有不太懂這個方法的可以查看我寫的筆記:https://segmentfault.com/n/13...

reduce() 方法接收一個函數(shù)作為累加器(accumulator),數(shù)組中的每個值(從左到右)開始縮減,最終為一個值。

callback (執(zhí)行數(shù)組中每個值的函數(shù),包含四個參數(shù))
initialValue (作為第一次調(diào)用 callback 的第一個參數(shù)。)

還有一個是Promise實現(xiàn)異步處理

有不太懂這個方法的可以查看我寫的筆記:https://segmentfault.com/n/13...

具體是使用Promise的這個方法:
Promise.resolve()
這個方法返回一個fulfilled的Promise實例,或者原始的Promise實例。

代碼實現(xiàn):

// 實現(xiàn)隊列 
// 本質(zhì): 對.then()方法實現(xiàn)累加 
let curPromise = urls.reduce((promise,curl) => {

    return promise.then(() => {
        return new Promise(resolve => {
            // 網(wǎng)絡獲取當前地址的網(wǎng)頁內(nèi)容
            requestGet(curl,() => {
                resolve(); 
            });
        });
    });

},Promise.resolve());
將數(shù)據(jù)寫入result.json文件中

代碼實現(xiàn):

// 寫入數(shù)據(jù)
curPromise.then(()=>{
    fs.writeFile("result.json", JSON.stringify(result), function (err) {
        if(err) throw new Error("appendFile failed...");
        console.log("數(shù)據(jù)寫入success...");
    });    
});
完整代碼
// 項目依賴
const cheerio = require("cheerio");
const agent = require("superagent");
const path = require("path");
const fs = require("fs");

// 地址數(shù)據(jù)
const urls = [{ 
    page:1,
    url:"https://www.imooc.com/course/list?c=fe&page=1"
},{
    page:2,
    url:"https://www.imooc.com/course/list?c=fe&page=2"
},{
    page:3,
    url:"https://www.imooc.com/course/list?c=fe&page=3"
}];

// 最終的數(shù)據(jù)
let result = [];


// 數(shù)據(jù)結(jié)構(gòu)
/**
 * [
 *     {
 *         page: 1,
 *         data: [
 *                {title:xx,imgurl:xx...},
 *                ......
 *               ]
 *     }
 *     ......    
 * ]
 */


// 發(fā)起get請求
function requestGet(urlObj,callback){

    agent.get(urlObj.url)
     .end((err,res) => {
         if(err) throw new Error(err);

         // 分析頁面
         let pageJson = analysis(res.text);

         // 拼接數(shù)據(jù)
         result.push({
             page:urlObj.page,
             data:pageJson
         });

         console.log(`寫入第${urlObj.page}頁的數(shù)據(jù)...`);

         // 執(zhí)行回調(diào)
         callback();
     
     });
}


// 對網(wǎng)頁分析
function analysis(data){

    let page = [];
    let $ = cheerio.load(data);
    let courseArr = $(".course-list").find(".course-card-container");
    courseArr.each((index,element) => {
        let _this = $(element);
        // 組裝數(shù)據(jù)
        page.push({
            title:_this.find(".course-card-name").text(),
            imgurl:path.join("http:",_this.find(".course-card-top img").attr("src")),
            level:_this.find(".course-card-info span:first-child").text(),
            // level:_this.find(".icon-set_sns").parent().prev().text(),
            studynum:_this.find(".icon-set_sns").parent().text(),
            description:_this.find(".course-card-desc").text()
        });
    });
    return page;
}


// 實現(xiàn)隊列 
// 本質(zhì): 對.then()方法實現(xiàn)累加 
let curPromise = urls.reduce((promise,curl) => {

    return promise.then(() => {
        return new Promise(resolve => {
            // 具體的內(nèi)容
            requestGet(curl,() => {
                resolve(); 
            });
        });
    });

},Promise.resolve());

// 寫入數(shù)據(jù)
curPromise.then(()=>{
    fs.writeFile("result.json", JSON.stringify(result), function (err) {
        if(err) throw new Error("appendFile failed...");
        console.log("數(shù)據(jù)寫入success...");
    });    
});
啟動項目
node app.js

可以看到終端有次序的輸出了以下內(nèi)容:

當打開生成的 result.json 文件,其結(jié)構(gòu)也符合我們的預期:

至此,這篇文章也就結(jié)束啦,如果您有好的想法請留言喲。

持續(xù)學習中...

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/93607.html

相關(guān)文章

  • 資源集 - 收藏集 - 掘金

    摘要:行爬取頂點全網(wǎng)任意小說掘金之前連續(xù)多篇文章介紹客戶端爬取平臺,今天我們從零開始,實現(xiàn)爬取頂點小說網(wǎng)任意一本小說的功能。文件標記所有文件我的后端書架后端掘金我的后端書架月前本書架主要針對后端開發(fā)與架構(gòu)。 30行js爬取頂點全網(wǎng)任意小說 - 掘金之前連續(xù)多篇文章介紹客戶端爬取平臺(dspider),今天我們從零開始,實現(xiàn)爬取頂點小說網(wǎng)任意一本小說的功能。 如果你還不知道客戶端爬取,可以先看...

    stdying 評論0 收藏0
  • 資源集 - 收藏集 - 掘金

    摘要:行爬取頂點全網(wǎng)任意小說掘金之前連續(xù)多篇文章介紹客戶端爬取平臺,今天我們從零開始,實現(xiàn)爬取頂點小說網(wǎng)任意一本小說的功能。文件標記所有文件我的后端書架后端掘金我的后端書架月前本書架主要針對后端開發(fā)與架構(gòu)。 30行js爬取頂點全網(wǎng)任意小說 - 掘金之前連續(xù)多篇文章介紹客戶端爬取平臺(dspider),今天我們從零開始,實現(xiàn)爬取頂點小說網(wǎng)任意一本小說的功能。 如果你還不知道客戶端爬取,可以先看...

    馬忠志 評論0 收藏0
  • 認識CSS樣式

    摘要:聲明在英文大括號中的的就是聲明,屬性和值之間用英文冒號分隔。嵌入式樣式慕課網(wǎng),超酷的互聯(lián)網(wǎng)技術(shù)免費學習平臺,創(chuàng)新的網(wǎng)絡一站式學習實踐體驗服務及時貼心,內(nèi)容專業(yè)有趣易學。6-2 css樣式的優(yōu)點   為什么使用css樣式來設置網(wǎng)頁的外觀樣式呢?右邊編輯器是一段文字,我們想把“超酷的互聯(lián)網(wǎng)”、“服務及時貼心”、“有趣易學”這...

    番茄西紅柿 評論0 收藏0
  • 零碎筆記:瀏覽器訪問一個網(wǎng)站所經(jīng)歷的步驟

    摘要:瀏覽器拿到了簡書網(wǎng)的完整的頁面代碼,在解析和渲染這個頁面的時候,里面的圖片靜態(tài)資源,他們同樣也是一個個請求都需要經(jīng)過上面的主要的七個步驟。瀏覽器根據(jù)拿到的資源對頁面進行渲染,最終把一個完整的頁面呈現(xiàn)給了用戶。 瀏覽器訪問一個網(wǎng)站所經(jīng)歷的步驟 Chrome搜索自身的DNS緩存 搜索操作系統(tǒng)自身的DNS緩存(瀏覽器沒有找到緩存或緩存已經(jīng)失效)查看Chrome瀏覽器的DNS緩存信息(chr...

    張金寶 評論0 收藏0
  • 課網(wǎng)_《Java消息中間件》學習總結(jié)

    摘要:時間年月日星期六說明本文部分內(nèi)容均來自慕課網(wǎng)。這個時候,可以啟動多臺積分系統(tǒng),來同時消費這個消息中間件里面的登錄消息,達到橫向擴展的作用。 時間:2017年07月22日星期六說明:本文部分內(nèi)容均來自慕課網(wǎng)。@慕課網(wǎng):http://www.imooc.com教學源碼:無學習源碼:https://github.com/zccodere/s... 第一章:課程介紹 1-1 課程安排 Java...

    twohappy 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<