node.js异步编程

释放双眼,带上耳机,听听看~!

1.Node.js异步编程

1.1 同步API, 异步API


1
2
3
4
5
6
7
8
9
10
1 // 路径拼接
2 const public = path.join(__dirname, 'public');
3 // 请求地址解析
4 const urlObj = url.parse(req.url);
5 // 读取文件
6 fs.readFile('./demo.txt', 'utf8', (err, result) => {
7     console.log(result);
8 });
9
10
  • 同步API:只有当前API执行完成后,才能继续执行下一个API


1
2
3
4
1console.log('before');
2console.log('after');
3
4

异步API:当前API的执行不会阻塞后续代码的执行


1
2
3
4
5
6
7
1console.log('before');
2setTimeout(
3   () => { console.log('last');
4}, 2000);
5console.log('after');
6
7

1.2 同步API, 异步API的区别( 获取返回值 )

  • 同步API可以从返回值中拿到API执行的结果, 但是异步API是不可以的


1
2
3
4
5
6
7
1    // 同步
2  function sum (n1, n2) {
3      return n1 + n2;
4  }
5  const result = sum (10, 20);
6
7

1
2
3
4
5
6
7
8
9
1    // 异步
2  function getMsg () {
3      setTimeout(function () {
4          return { msg: 'Hello Node.js' }
5      }, 2000);
6  }
7  const msg = getMsg ();
8
9

1.3 回调函数

  • 自己定义函数让别人去调用。


1
2
3
4
5
6
1  // getData函数定义
2 function getData (callback) {}
3  // getData函数调用
4 getData (() => {});
5
6

1.4 使用回调函数获取异步API执行结果


1
2
3
4
5
6
7
8
9
10
1function getMsg (callback) {
2    setTimeout(function () {
3        callback ({ msg: 'Hello Node.js' })
4    }, 2000);
5}
6getMsg (function (msg) {
7    console.log(msg);
8});
9
10

1.5 同步API, 异步API的区别(代码执行顺序)

  • 同步API从上到下依次执行,前面代码会阻塞后面代码的执行


1
2
3
4
5
6
1for (var i = 0; i < 100000; i++) {
2    console.log(i);
3}
4console.log('for循环后面的代码');
5
6
  • 异步API不会等待API执行完成后再向下执行代码


1
2
3
4
5
6
1console.log('代码开始执行');
2setTimeout(() => { console.log('2秒后执行的代码')}, 2000);
3setTimeout(() => { console.log('"0秒"后执行的代码')}, 0);
4console.log('代码结束执行');
5
6

1.6 代码执行顺序分析


1
2
3
4
5
6
7
8
9
10
1console.log('代码开始执行');
2setTimeout(() => {
3    console.log('2秒后执行的代码');
4}, 2000);
5setTimeout(() => {
6    console.log('"0秒"后执行的代码');
7}, 0);
8console.log('代码结束执行');
9
10

node.js异步编程

1.7 Node.js中的异步API


1
2
3
1 fs.readFile('./demo.txt', (err, result) => {});
2
3

1
2
3
4
5
1
2var server = http.createServer();
3server.on('request', (req, res) => {});
4
5
  • 如果异步API后面代码的执行依赖当前异步API的执行结果,但实际上后续代码在执行的时候异步API还没有返回结果,这个问题要怎么解决呢?


1
2
3
4
1fs.readFile('./demo.txt', (err, result) => {});
2console.log('文件读取结果');
3
4

1.8 Promise

  • Promise出现的目的是解决Node.js异步编程中回调地域的问题。


1
2
3
4
5
6
7
8
9
10
11
12
13
1let promise = new Promise((resolve, reject) => {
2    setTimeout(() => {
3        if (true) {
4            resolve({name: '张三'})
5        }else {
6            reject('失败了')
7        }
8    }, 2000);
9});
10promise.then(result => console.log(result); // {name: '张三'})
11       .catch(error => console.log(error); // 失败了)
12
13

1.9 异步函数

  • 异步函数是异步编程语法的终极解决方案,它可以让我们将异步代码写成同步的形式,让代码不再有回调函数嵌套,使代码变得清晰明了。


1
2
3
1const fn = async () => {};
2
3

1
2
3
1async function fn () {}
2
3

async关键字

  1. 普通函数定义前加async关键字 普通函数变成异步函数
  2. 异步函数默认返回promise对象
  3. 在异步函数内部使用return关键字进行结果返回 结果会被包裹的promise对象中 return关键字代替了resolve方法
  4. 在异步函数内部使用throw关键字抛出程序异常
  5. 调用异步函数再链式调用then方法获取异步函数执行结果
  6. 调用异步函数再链式调用catch方法获取异步函数执行的错误信息

await关键字

  1. await关键字只能出现在异步函数中
  2. await promise await后面只能写promise对象 写其他类型的API是不不可以的
  3. await关键字可是暂停异步函数向下执行 直到promise返回结果
  • 需求:依次读取A文件、B文件、C文件


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
1const fs = require('fs');
2// 改造现有异步函数api 让其返回promise对象 从而支持异步函数语法
3const promisify = require('util').promisify;
4// 调用promisify方法改造现有异步API 让其返回promise对象
5const readFile = promisify(fs.readFile);
6
7async function run () {
8   let r1 = await readFile('./1.txt', 'utf8')
9   let r2 = await readFile('./2.txt', 'utf8')
10  let r3 = await readFile('./3.txt', 'utf8')
11  console.log(r1)
12  console.log(r2)
13  console.log(r3)
14}
15
16run();
17
18

给TA打赏
共{{data.count}}人
人已打赏
安全技术

DES 加密 解密

2021-8-18 16:36:11

安全技术

C++ 高性能服务器网络框架设计细节

2022-1-11 12:36:11

个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索