Skip to content

Commit

Permalink
Finished
Browse files Browse the repository at this point in the history
  • Loading branch information
andrew committed Oct 31, 2014
1 parent 431937a commit 93f499b
Showing 1 changed file with 6 additions and 83 deletions.
89 changes: 6 additions & 83 deletions lesson17/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,28 +2,13 @@

## 目标

建立一个 lesson5 项目,在其中编写代码。
理解promise的三种状态

代码的入口是 `app.js`,当调用 `node app.js` 时,它会输出 CNode(https://cnodejs.org/ ) 社区首页的所有主题的标题,链接和第一条评论,以 json 的格式。
学会promise的基本api: then, Q.all, Q.allSettled

注意:与上节课不同,并发连接数需要控制在 5 个。
学会用promise来替换回调函数

输出示例:

```js
[
{
"title": "【公告】发招聘帖的同学留意一下这里",
"href": "http://cnodejs.org/topic/541ed2d05e28155f24676a12",
"comment1": "呵呵呵呵"
},
{
"title": "发布一款 Sublime Text 下的 JavaScript 语法高亮插件",
"href": "http://cnodejs.org/topic/54207e2efffeb6de3d61f68f",
"comment1": "沙发!"
}
]
```
学会编写自己的promise链

## 知识点

Expand Down Expand Up @@ -698,77 +683,15 @@ Q.allSettled([printFileContent('nosuchfile.txt'),printFileContent('sample02.txt'
.done();

```
这次我们要介绍的是 async 的 `mapLimit(arr, limit, iterator, callback)` 接口。另外,还有个常用的控制并发连接数的接口是 `queue(worker, concurrency)`,大家可以去 https://github.com/caolan/async#queueworker-concurrency 看看说明。

这回我就不带大家爬网站了,我们来专注知识点:并发连接数控制。

对了,还有个问题是,什么时候用 eventproxy,什么时候使用 async 呢?它们不都是用来做异步流程控制的吗?

我的答案是:

当你需要去多个源(一般是小于 10 个)汇总数据的时候,用 eventproxy 方便;当你需要用到队列,需要控制并发数,或者你喜欢函数式编程思维时,使用 async。大部分场景是前者,所以我个人大部分时间是用 eventproxy 的。

正题开始。

首先,我们伪造一个 `fetchUrl(url, callback)` 函数,这个函数的作用就是,当你通过

```js
fetchUrl('http://www.baidu.com', function (err, content) {
// do something with `content`
});
```

调用它时,它会返回 `http://www.baidu.com` 的页面内容回来。

当然,我们这里的返回内容是假的,返回延时是随机的。并且在它被调用时,会告诉你它现在一共被多少个地方并发调用着。

```js
// 并发连接数的计数器
var concurrencyCount = 0;
var fetchUrl = function (url, callback) {
// delay 的值在 2000 以内,是个随机的整数
var delay = parseInt((Math.random() * 10000000) % 2000, 10);
concurrencyCount++;
console.log('现在的并发数是', concurrencyCount, ',正在抓取的是', url, ',耗时' + delay + '毫秒');
setTimeout(function () {
concurrencyCount--;
callback(null, url + ' html content');
}, delay);
};
```

我们接着来伪造一组链接

```js
var urls = [];
for(var i = 0; i < 30; i++) {
urls.push('http://datasource_' + i);
}
```

这组链接的长这样:

![](https://raw.githubusercontent.com/alsotang/node-lessons/master/lesson5/1.png)

接着,我们使用 `async.mapLimit` 来并发抓取,并获取结果。

```js
async.mapLimit(urls, 5, function (url, callback) {
fetchUrl(url, callback);
}, function (err, result) {
console.log('final:');
console.log(result);
});
```
运行输出是这样的:

![](https://raw.githubusercontent.com/alsotang/node-lessons/master/lesson5/2.png)

可以看到,一开始,并发链接数是从 1 开始增长的,增长到 5 时,就不再增加。当其中有任务完成时,再继续抓取。并发连接数始终控制在 5 个。

完整代码请参见 app.js 文件。
## 结束语
当你理解完上面所有的知识点时,你就会正确高效的使用promise了。本节只是讲了promise的原理和几个基本的API,不过你掌握了这些之后,再去看q的文档,应该很容易就能理解各个api的意图。

0 comments on commit 93f499b

Please sign in to comment.