探索Javascript 异步编程
在我们日常编码中,需要异步的场景很多,比如读取文件内容、获取远程数据、发送数据到服务端等。因为浏览器环境里Javascript是单线程的,所以异步编程在前端领域尤为重要。
异步的概念
所谓异步,是指当一个过程调用发出后,调用者不能立刻得到结果。实际处理这个调用的过程在完成后,通过状态、通知或者回调来通知调用者。
比如我们写这篇文字时点击发布按钮,我们并不能马上得到文章发布成功或者失败。等待服务器处理,这段时间我们可以做其他的事情,当服务器处理完成后,通知我们是否发布成功。
所谓同步,是指当一个过程调用发出后,必须等待这个过程处理完成后,再处理其他事情。即堵塞执行。
异步的方式
在es6之前,我们实现异步有4种方法,回调、事件、发布订阅和promise方式。
异步之回调:
function dealTask(param, callback) {
// Deal with some time consuming tasks.
// ...
Object.prototype.toString.call(callback) === '[object Function]' ? callback() : null;
}
dealTask({ id: 1 }, function() {
console.log('... I am in the callback...');
})
回调的方式来实现异步其实就是把需要在当前任务完成后执行的函数当成参数传入,完成任务后执行即可。
异步之事件
function dealTask(param) {
// Deal with some time consuming tasks.
// ...
events.trigger('dealTaskFinish')
}
events.on('dealTaskFinish', function() {
console.log('...I am in the end...');
})
通过事件来实现回调,好处是方便实用,跨模块传递数据。坏处是,事件用的多了后业务逻辑混乱,不知道哪里注册过哪里监听过。
另外需要注意的是在web component场景下,mount后注册过的事件需要在unmount释放,不然会导致内存泄露。
异步之发布订阅
发布订阅的简单例子是,一个开关,同时并联几个灯泡(在不同房间),触发的时候,几个灯泡都会得到指令,然后执行发光的行为。
// 使用pubsubz实现
var testSubscriber = function(data ){
console.log(data );
};
var testSubscription = pubsubz.subscribe( 'example', testSubscriber );
pubsubz.publish( 'example', 'hello' );
订阅发布与的性质与"事件监听"类似,不同的是,我们可以通过查看"消息中心",了解存在多少信号、每个信号有多少订阅者,从而监控程序的运行。
异步之Promise
function helloWorld (ready) {
return new Promise(function (resolve, reject) {
if (ready) {
resolve("Hello World!")
} else {
reject("Good bye!")
}
})
}
helloWorld(true).then(function (message) {
console.log(message)
}, function (error) {
console.log(error)
})
Promises对象是CommonJS工作组提出的一种规范,是对异步编程的一种统一,其实也就是语法糖,可阅读性变强了而已。
在ES6出来以后,我们的异步方式也发生了改变。
异步之Generator
Generator函数是协程在ES6的实现,最大特点就是可以交出函数的执行权(即暂停执行)。
Generator函数可以暂停执行和恢复执行,这是它能封装异步任务的根本原因。除此之外,它还有两个特性,使它可以作为异步编程的完整解决方案:函数体内外的数据交换和错误处理机制。
function* Foo(x) {
yield x + 1;
var y = yield null;
return x + y;
}
var foo = Foo(5);
foo.next(); // { value: 6, done: false }
foo.next(); // { value: null, done: false }
foo.next(8); // { value: 13, done: true }
next方法返回值的value属性,是Generator函数向外输出数据;next方法还可以接受参数,这是向Generator函数体内输入数据。
yield命令用于将程序的执行权移出Generator函数,那么就需要一种方法,将执行权再交还给Generator函数
上面的方式,是我们手动调用Generator函数执行,但是当我们的需要执行next方法很多时,就需要Generator函数自动执行了。
Generator函数自动执行的意思是,通过一定的方法来自动执行next方法,比如:
function autoRunGen(gen){
var g = gen();
function next(data){
var result = g.next(data);
if (result.done) return result.value;
result.value.then(function(data){
next(data);
});
}
next();
}
co模块是TJ开发的一个小工具,用于Generator函数的自动执行。他的主要思想和上面的代码片段类似。
使用co的前提条件是,Generator函数的yield命令后面,只能是Promise对象。
co(function *() {
var data = yield $.get('/api/data');
console.log(data);
var user = yield $.get('/api/user');
console.log(user);
var products = yield $.get('/api/products');
console.log(products);
});
co模块使得我们可以像写同步代码一样,写异步代码。
异步之async/await
async函数仅仅是Generator函数的语法糖。
var fs = require('fs');
var readFile = function (fileName) {
return new Promise(function (resolve, reject) {
fs.readFile(fileName, function(error, data) {
if (error) reject(error);
resolve(data);
});
});
};
var gen = function* (){
var f1 = yield readFile('/etc/a.js');
var f2 = yield readFile('/etc/b.js');
console.log(f1.toString());
console.log(f2.toString());
};
使用async/await方式:
var asyncReadFile = async function (){
var f1 = await readFile('/etc/a.js');
var f2 = await readFile('/etc/b.js');
console.log(f1.toString());
console.log(f2.toString());
};
async函数就是将Generator函数的星号(*)替换成async,将yield替换成await,仅此而已。
不同的是,Generator执行需要手动执行,而async函数可以自动执行,像写同步一样写异步。Generator返回Iterator对象,async函数返回Promise对象。
探索Javascript 异步编程的更多相关文章
- 探索Javascript异步编程
异步编程带来的问题在客户端Javascript中并不明显,但随着服务器端Javascript越来越广的被使用,大量的异步IO操作使得该问题变得明显.许多不同的方法都可以解决这个问题,本文讨论了一些方法 ...
- JavaScript异步编程(2)- 先驱者:jsDeferred
JavaScript当前有众多实现异步编程的方式,最为耀眼的就是ECMAScript 6规范中的Promise对象,它来自于CommonJS小组的努力:Promise/A+规范. 研究javascri ...
- JavaScript异步编程的主要解决方案—对不起,我和你不在同一个频率上
众所周知(这也忒夸张了吧?),Javascript通过事件驱动机制,在单线程模型下,以异步的形式来实现非阻塞的IO操作.这种模式使得JavaScript在处理事务时非常高效,但这带来了很多问题,比如异 ...
- JavaScript异步编程原理
众所周知,JavaScript 的执行环境是单线程的,所谓的单线程就是一次只能完成一个任务,其任务的调度方式就是排队,这就和火车站洗手间门口的等待一样,前面的那个人没有搞定,你就只能站在后面排队等着. ...
- javascript异步编程的前世今生,从onclick到await/async
javascript与异步编程 为了避免资源管理等复杂性的问题, javascript被设计为单线程的语言,即使有了html5 worker,也不能直接访问dom. javascript 设计之初是为 ...
- Promises与Javascript异步编程
Promises与Javascript异步编程 转载:http://www.zawaliang.com/2013/08/399.html 在如今都追求用户体验的时代,Ajax应用真的是无所不在.加上这 ...
- 5分种让你了解javascript异步编程的前世今生,从onclick到await/async
javascript与异步编程 为了避免资源管理等复杂性的问题,javascript被设计为单线程的语言,即使有了html5 worker,也不能直接访问dom. javascript 设计之初是 ...
- 转: Promises与Javascript异步编程
在如今都追求用户体验的时代,Ajax应用真的是无所不在.加上这些年浏览器技术.HTML5以及CSS3等的发展,越来越多的富Web应用出现:在给与我们良好体验的同时,Web开发人员在背后需要处理越来越多 ...
- JavaScript异步编程
前言 如果你有志于成为一个优秀的前端工程师,或是想要深入学习JavaScript,异步编程是必不可少的一个知识点,这也是区分初级,中级或高级前端的依据之一.如果你对异步编程没有太清晰的概念,那么我建议 ...
随机推荐
- sql查询原理和Select执行顺序
一 sql语句的执行步骤 1)语法分析,分析语句的语法是否符合规范,衡量语句中各表达式的意义. 2) 语义分析,检查语句中涉及的所有数据库对象是否存在,且用户有相应的权限. 3)视图转换,将涉及视图的 ...
- String与其他类型的转换
首先,对于String类有一点是毫无疑问的:对String对象的任何改变都不影响到原对象,相关的任何change操作都会生成新的对象. 一. String与StringBuilder.StringBu ...
- JS 原生JS 判断滚动条滑动到底部
window.addEventListener("scroll", function(event) { var scrollTop = document.documentEleme ...
- linux命令(6/10):find 命令
Linux将时钟分为系统时钟(System Clock)和硬件(Real Time Clock,简称RTC)时钟两种.系统时间是指当前Linux Kernel中的时钟, 而硬件时钟则是主板上由电池供电 ...
- 华为S5700系列交换机配置通过Telnet登录设备
声明:不管什么服务,都需要交换机开启服务,创建对应权限的用户,在通道下允许协议通过,缺一不可,以telnet为例. 组网图形 图1 配置通过Telnet登录设备组网图 组网需求 如图一所示,PC与设备 ...
- Linux下针对路由功能配置iptables的方法详解
作为公司上网的路由器需要实现的功能有nat地址转换.dhcp.dns缓存.流量控制.应用程序控制,nat地址转换通过iptables可以直 接实现,dhcp服务需要安装dhcpd,dns缓存功能需要使 ...
- lamp架构之升级php版本
当你看到这篇文章的时候 YHSPY.COM 服务器上的PHP版本已经从 5.4.27 升级到了 7.0.4,这是一个重大的飞跃.一路升级遇到了很多问题.官方声称PHP7最大的升级就是在语言性能上的提升 ...
- Synchronize原理
1 普通方法上 2 静态方法上 修饰静态方法内置锁是当前的Class字节码对象 修饰普通方法内置锁是当前类的实例 原理与使用: 从字节码层面解释: 执行同步代码块 monitorenter synch ...
- Elasticsearch6.4.3文档的映射
已经把ElasticSearch的核心概念和关系数据库做了一个对比,索引(index)相当于数据库,类型(type)相当于数据表,映射(Mapping)相当于数据表的表结构.ElasticSearch ...
- 文件(2)--IO流
IO流 输入流和输出流 Java中的IO流根据功能划分为:输入流和输出流.输入流:用于读取数据.输出流:用于写出数据.输入输出的参照方向是根据我们的程序的. 字节流和字符流 Java中的IO流根据处理 ...