在nodeschool上学习时遇到的问题:learnyounode里的第九题:
写一个小程序,要求输入三个参数url,使用该程序模拟三次get请求分别从这三个url获取数据,全部返回后打印到控制台上。要求:打印出的响应数据要和三个url参数的顺序一样,即打印出的第一条应该是第一个url返回的数据,url由命令行参数提供(process.argv)。
因为node是异步的,所以不知道应该如何控制这个输出的顺序,求帮忙!
学习是最好的投资!
原生版,用results和counter兩個變數維護狀態,保持順序靠那個idx變數實作
jsvar http = require('http'); var args = process.argv; if(args.shift() === 'node') args.shift(); if(process.argv.length !== 3) { console.log('must input 3 url, got', args); process.exit(); } var results = []; var counter = 0; args.forEach(function(url, idx) { http.get(url, function(res) { res.on('data', function (chunk) { if(!results[idx]) { results[idx] = ''; } results[idx] += chunk; }).on('end', function() { counter++; if (counter === args.length) { results.forEach(function(res, i) { console.log(args[i], res); }); } }); }).on('error', function(e) { throw e; }); });
js
var http = require('http'); var args = process.argv; if(args.shift() === 'node') args.shift(); if(process.argv.length !== 3) { console.log('must input 3 url, got', args); process.exit(); } var results = []; var counter = 0; args.forEach(function(url, idx) { http.get(url, function(res) { res.on('data', function (chunk) { if(!results[idx]) { results[idx] = ''; } results[idx] += chunk; }).on('end', function() { counter++; if (counter === args.length) { results.forEach(function(res, i) { console.log(args[i], res); }); } }); }).on('error', function(e) { throw e; }); });
能夠完成需求,但容錯、擴充、可讀性一塌糊塗
request+bluebird版
var request = require('request'); var Promise = require('bluebird'); var prequest = Promise.promisify(request); var args = process.argv; if(args.shift() === 'node') args.shift(); if(process.argv.length !== 3) { console.log('must input 3 url, got', args); process.exit(); } Promise.all(args.map(function(url) { return prequest(url); })) .each(function(result) { var response = result[0]; console.log(response.request.href, response.body); }) .catch(function(e) { console.error(e); });
題目沒說並行請求還是串行,這裡都是並行的,串行的話反正天然保持順序閉著眼睛打印就行
雷雷
一般以前我碰上這種情況都會果斷的選擇回調來寫的,程式碼大概是下面這個樣子:
var http = require("http"), urls = process.argv, datas = []; if( urls.shift() === "node" ) urls.shift(); function getHTML( url, callback ) { http.get( url, function( res ) { var result = ""; res.on( "data", function(chunk) { result += chunk; }) .on( "end", function() { callback( result ); }); }) } (function get( p ) { if( p === urls.length ) return last(); getHTML( urls[p], function(res) { datas.push( res ) get( ++p ); }) function last() { datas.forEach(function(d){ console.log(d) }) } })(0);
但自從接觸了 Promise 之後感覺這樣寫異步要順暢多了,所以慢慢的也轉變過來了,代碼大概是這個樣子的:
var Promise = require("promise").Promise, http = require("http"); var urls = process.argv; if( urls.shift() === "node" ) urls.shift(); function getHTML( url ) { return new Promise(function( resolve, reject ) { http.get( url, function( res ) { var result = ""; res.on( "data", function(chunk) { result += chunk }) .on( "end", function() { resolve(result) }) .on( "error", function() { reject(result) }) }) }) } urls.map(getHTML).reduce(function(seq, html) { return seq.then(function(){return html}).then(function(res){ console.log( res ); }) }, Promise.resolve());
原生版,用results和counter兩個變數維護狀態,保持順序靠那個idx變數實作
能夠完成需求,但容錯、擴充、可讀性一塌糊塗
request+bluebird版
題目沒說並行請求還是串行,這裡都是並行的,串行的話反正天然保持順序閉著眼睛打印就行
雷雷
雷雷
一般以前我碰上這種情況都會果斷的選擇回調來寫的,程式碼大概是下面這個樣子:
但自從接觸了 Promise 之後感覺這樣寫異步要順暢多了,所以慢慢的也轉變過來了,代碼大概是這個樣子的: