Ndejs中的异步编程
同步API,异步API
同步API:只有当前API执行完成后,才能继续执行下一个API
console.log('before');
console.log('after');
异步API:当前API的执行不会阻碍后续代码的执行
console.log('before');
setTimeout(
() => { console.log('last');
}, 2000);
console.log('after');
代码设计模式
异步编程有很多特有的代码设计模式,为了实现同样的功能,使用同步方式和异步方式编写代码会有很大差异,以下举例。
1、函数返回值
// 同步
function sum (n1, n2) {
return n1 + n2;
}
const result = sum (10, 20);
// 异步
function getMsg () {
setTimeout(function () {
return { msg: 'Hello Node.js' }
}, 2000);
}
const msg = getMsg ();
在异步方式下,由于函数执行结果不是通过返回值,而是通过回调函数传递,一般按以下方式编写代码:
fn2('input', function(output2) {
fn1(output2, function(output1) {
//Do something.
});
});
2.代码执行顺序
同步API从上到下依次执行,前面代码会阻塞后面代码的执行
for (var i = 0; i < 100000; i++) {
console.log(i);
}
console.log('for循环后面的代码');
异步API不会等待API执行完成后再向下执行代码
console.log('代码开始执行');
setTimeout(() => { console.log('2秒后执行的代码')},
2000);
setTimeout(() => { console.log('"0秒"后执行的代码')},
0);
console.log('代码结束执行');
分析:
- 按照顺序从上到下的执行代码
- 遇到同步执行代码直接执行
- 遇到异步代码,放到异步代码执行区
- 所有的同步代码执行完毕之后,再开始按照顺序执行异步区域的代码
回调
Promise
按照顺序读取文件
新建文本 1.txt 2.txt 3.txt
写入内容
const fs=require('fs')
function p1(){
return new Promise((resolve,reject)=>{
fs.readFile('1.txt','utf8',(err,data)=>{
resolve(data)
})
})
}
function p2(){
return new Promise((resolve,reject)=>{
fs.readFile('2.txt','utf8',(err,data)=>{
resolve(data)
})
})
}
function p3(){
return new Promise((resolve,reject)=>{
fs.readFile('3.txt','utf8',(err,data)=>{
resolve(data)
})
})
}
p1().then(result=>{
console.log(result)
return p2()
}).then(result=>{
console.log(result)
return p3()
}).then(result=>{
console.log(result)
return p2()
})