[Effective JavaScript 笔记]第62条:在异步序列中使用嵌套或命名的回调函数
异步程序的操作顺序
61条讲述了异步API如何执行潜在的代价高昂的I/O操作,而不阻塞应用程序继续处理其他输入。理解异步程序的操作顺序刚开始有点混乱。例如,下面的代码会在打印"finished"之前打印“starting”,即使这两个动作的程序源文件中以相反的顺序呈现。
downloadAsync('file.txt',function(file){
console.log('finished');
});
console.log('starting');
downloadAsync调用会立即返回,不会等待文件完成下载。同时,js的运行到完成机制确保下一行代码会在处理其他事件处理程序之前被执行。也就是说"starting"一定会在"finished"之前被打印。
理解操作序列的最简单的方式是异步API是发起操作而不是执行操作。上面的代码发起了一个文件的下载然后立即打印了“starting”。当下载完成后,在事件循环的某个单独的轮次中,被注册的事件处理程序才会打印出“finished”。
如何串联异步操作
如果你需要在发起一个操作后做一些事情,如果只能在一行中放置好几个声明,那么如何串联已完成的异步操作呢?例如,如果我们需要在异步数据库中查找一个URL,然后下载这个URL的内容?不可能发起两个连续的请求。
db.lookupAsync('url',function(url){
});
downloadAsync(url,function(text){//error:url is bound
console.log('contents of ' + url +':'+text);
});
以上代码不可能工作,因为从数据库查询到的URL结果需要作为downloadAsync方法的参数。但是它并不在作用域内。我们所做的这一步只是发起数据库查找,查找的结果还不可用。
回调函数处理
最简单的处理方法使用嵌套。借助于闭包的魔力,将第二个动作嵌套在第一个动作的回调函数中。
db.lookupAsync('url',function(url){
downloadAsync(url,function(text){
console.log('contents of ' + url +':'+text);
});
})
这里有两个回调函数,但第二个被包含在第一个中,创建闭包能够访问外部回调函数的变量。
嵌套的异步操作很容易,但当扩展到更长的序列时会很快变得麻烦。
db.lookupAsync('url',function(url){
downloadAsync(url,function(file){
downloadAsync('a.txt',function(a){
downloadAsync('b.txt',function(b){
downloadAsync('c.txt',function(c){
//....
});
});
});
});
});
回调命名的函数
减少过多的嵌套的方法之一是将嵌套的回调函数作为命名的函数,并将它们需要附加数据作为额外的参数传递。以上代码可以改写为:
db.lookupAsync('url',downloadURL);
function downloadURL(url){
downloadAsync(url,function(text){
showContents(url,text);
});
}
function showContents(url,text){
console.log('contents of ' + url +':'+text);
}
使用bind方法消除嵌套
为了合并外部的url变量和内部的text变量作为showContents方法的参数,在downloadURL方法中仍然使用了嵌套的回调函数。这里可以使用bind方法消除最深层的嵌套回调函数。
db.lookupAsync('url',downloadURL);
function downloadURL(url){
downloadAsync(url,showContents.bind(null,url));
}
function showContents(url,text){
console.log('contents of ' + url +':'+text);
}
这种做法可以使代码看起来很有顺序性,但需要为操作序列的每个中间步骤命名,并且一步步地使用绑定。这可能导致尴尬的情况,如多层嵌套时。
db.lookupAsync('url',downloadURLAndFiles);
function downloadURLAndFiles(url){
downloadAsync(url,downloadABC.bind(null,url));
}
function downloadABC(url,file){
downloadAsync('a.txt',downloadBC.bind(null,url,file));
}
function downloadBC(url,file,a){
downloadAsync('b.txt',downloadC.bind(null,url,file,a));
}
function downloadC(url,file,a,b){
downloadAsync('c.txt',finish.bind(null,url,file,a,b));
}
function finish(url,file,a,b,c){
//....
}
结合两种方法
结合这两种方法,会使代码更易理解。
db.lookupAsync('url',function(url){
downloadURLAndFiles(url);
});
function downloadURLAndFiles(url){
downloadAsync(url,downloadFiles.bind(null,url));
}
function downloadFiles(url,file){
downloadAsync('a.txt',function(a){
downloadAsync('b.txt',function(b){
downloadAsync('c.txt',function(c){
//...
});
});
});
}
最后一步可以使用一个额外的抽象来简化,可以下载多个文件并将它们存储在数组中。
function downloadFiles(url,file){
downloadAllAsync(['a.txt','b.txt','c.txt'],function(all){
var a=all[0],b=all[1],c=all[2];
});
}
使用downloadAllAsync函数允许我们同时下载多个文件。排序意味着每个操作只有等前一个操作完成后才能启动。一些操作本质上是连续的,比如下载我们从数据库查询到的URL。但如果我们有一个文件列表要下载,没理由等每个文件完成下载后才请求接下来的一个。
除了嵌套和命名回调,还可以建立更高层的抽象使异步控制流更简单、更简洁。
提示
使用嵌套或命名的回调函数按顺序地执行多个异步操作
尝试在过多的嵌套的回调函数和尴尬的命名的非嵌套回调函数之间取得平衡
避免将可被并行执行的操作顺序化
[Effective JavaScript 笔记]第62条:在异步序列中使用嵌套或命名的回调函数的更多相关文章
- [Effective JavaScript 笔记]第21条:使用apply方法通过不同数量的参数调用函数
apply()方法定义 函数的apply()方法和call方法作用相同,区别在于接收的参数的方式不同.apply()方法接收两个参数,一个是对象,一个是参数数组. apply()作用 1.用于延长函数 ...
- [Effective JavaScript 笔记]第34条:在原型中存储方法
js中完全有可能不借助原型进行编程.不用在其原型中定义任何的方法. 创建对象 构造函数法 所有属性和方法都在构造函数中定义 function User(name,pwd){ this.name=nam ...
- [Effective JavaScript 笔记]第27条:使用闭包而不是字符串来封装代码
函数是一种将代码作为数据结构存储的便利方式,代码之后可以被执行.这使得富有表现力的高阶函数抽象如map和forEach成为可能.它也是js异步I/O方法的核心.与此同时,也可以将代码表示为字符串的形式 ...
- [Effective JavaScript 笔记] 第4条:原始类型优于封闭对象
js有5种原始值类型:布尔值.数字.字符串.null和undefined. 用typeof检测一下: typeof true; //"boolean" typeof 2; //&q ...
- [Effective JavaScript 笔记] 第5条:避免对混合类型使用==运算符
“1.0e0”=={valueOf:function(){return true;}} 是值是多少? 这两个完全不同的值使用==运算符是相等的.为什么呢?请看<[Effective JavaSc ...
- [Effective JavaScript 笔记]第28条:不要信赖函数对象的toString方法
js函数有一个非凡的特性,即将其源代码重现为字符串的能力. (function(x){ return x+1 }).toString();//"function (x){ return x+ ...
- [Effective JavaScript 笔记]第68条:使用promise模式清洁异步逻辑
构建异步API的一种流行的替代方式是使用promise(有时也被称为deferred或future)模式.已经在本章讨论过的异步API使用回调函数作为参数. downloadAsync('file.t ...
- [Effective JavaScript 笔记]第64条:对异步循环使用递归
假设需要有这样一个函数,接收一个URL的数组并尝试依次下载每个文件直到有一个文件被成功下载.如果API是同步的,使用循环很简单实现. function downloadOneSync(urls){ f ...
- [Effective JavaScript 笔记]第67条:绝不要同步地调用异步的回调函数
设想有downloadAsync函数的一种变种,它持有一个缓存(实现为一个Dict)来避免多次下载同一个文件.在文件已经被缓存的情况下,立即调用回调函数是最优选择. var cache=new Dic ...
随机推荐
- 【MPI0】学习资料搜集
一个不错的英文的MPI教程:http://mpitutorial.com 中科大的MPI学习资料:http://micro.ustc.edu.cn/Linux/MPI/ 清华大学的并行计算资料:htt ...
- [poj3046]Ant Counting(母函数)
题意: S<=x1+x2+...+xT<=B 0<=x1<=N1 0<=x2<=N2 ... 0<=xT<=NT 求这个不等式方程组的解的个数. 分析: ...
- 用wcf实现带有“秒传”功能的网盘
写在前面 前面记录过这样一个关于“秒传”的实现思路,在这篇就弄了一个简单的demo实现了一下,当中有很多业务仍没考虑,只是将“秒传”的实现思路,用代码实现了一下. 关于秒传,可以参考这篇文章:何为“秒 ...
- WordPress 插件机制的简单用法和原理(Hook 钩子)
WordPress 的插件机制实际上只的就是这个 Hook 了,它中文被翻译成钩子,允许你参与 WordPress 核心的运行,是一个非常棒的东西,下面我们来详细了解一下它. PS:本文只是简单的总结 ...
- sql-函数avg,count,max,min,sum
常用函数 AVG (平均) COUNT (计数) MAX (最大值) MIN (最小值) SUM (总合) 运用函数的语法是: SELECT "函数名"("栏位名&qu ...
- array,vertor,arraylist,hashable,hashmap等几个易混淆概念的区别
Array可以存放Object和基本数据类型,但创建时必须指定数组的大小,并不能再改变, Vertor是放的Object. Vertor一维,Hashmap/Hashtabe二维: Vertor/Ar ...
- 【POJ 2096】Collecting Bugs 概率期望dp
题意 有s个系统,n种bug,小明每天找出一个bug,可能是任意一个系统的,可能是任意一种bug,即是某一系统的bug概率是1/s,是某一种bug概率是1/n. 求他找到s个系统的bug,n种bug, ...
- Yii2提示信息设置方法
显示信息提示用户时,可以用setFlash,hasFlash,getFlash function actionOk() { Yii::app()->user->setFlash('succ ...
- 洛谷P2024 食物链
挺神奇 题目描述 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A 吃 B,B 吃 C,C 吃 A. 现有 N 个动物,以 1 - N 编号.每个动物都是 A,B,C 中的一种 ...
- 网页设计师常用的PHOTOSHOP插件
Photoshop是网页设计师常用的一个非常重要而强大的工具,可以让网页设计师的工作高效便捷的进行,也为设计师们的天马行空提供了实际技术实现.一般我们的网页设计师设计完成后,需要将其转换制作成网页形式 ...