Node-异步编程

Posted 我真的爱敲代码

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Node-异步编程相关的知识,希望对你有一定的参考价值。


Node.js异步编程

同步API和异步API

 // 路径拼接
 const public = path.join(__dirname, 'public');
 // 请求地址解析
 const urlObj = url.parse(req.url);
 // 读取文件
 fs.readFile('./demo.txt', 'utf8', (err, result) => {
     console.log(result);
 });

同步API:只有当前API执行完成后,才能继续执行下一个API

console.log('before'); 
console.log('after');

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

console.log('before');
setTimeout(
   () => { console.log('last');
}, 2000);
console.log('after');

同步API和异步API区别(获取返回值):
同步API可以从返回值中拿到API执行的结果, 但是异步API是不可以的

    // 同步
  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 ();
  console.log(msg);   //undefined

回调函数

自己定义函数让别人调用

  // getData函数定义,callback所对应的函数就是回调函数
 function getData (callback) {}
  // getData函数调用
 getData (() => {});

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

function getMsg (callback) {
    setTimeout(function () {
        callback ({ msg: 'Hello Node.js' })
    }, 2000);
}
getMsg (function (msg) {    //callback是一个形参,对应的实参是一个匿名函数
    console.log(msg);
});

同步API,异步API的区别(代码执行顺序):
同步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('代码结束执行');

代码执行顺序分析

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

node.js先执行所有同步API,然后再去执行异步代码

Node.js中的异步API

fs.readFile('./demo.txt', (err, result) => {});
 var server = http.createServer();
 server.on('request', (req, res) => {});

如果异步API后面代码的执行依赖当前异步API的执行结果,但实际上后续代码在执行的时候异步API还没有返回结果,这个问题要怎么解决呢?

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

需求:依次读取A文件、B文件、C文件,若都使用回调函数(将C文件放在B文件的回调函数中,将B文件放在A文件的回调函数中),会导致代码嵌套过多。

const fs = require('fs');

fs.readFile('./1.txt','utf8',(err,result1) => {
	console.log(result1);
	fs.readFile('./2.txt','utf8',(err,result2) => {
		console.log(result2);
		fs.readFile('./3.txt','utf8',(err,result3) => {
			console.log(result3);
		})
	})
});

Promise

回调函数:当一个函数作为参数传入另一个函数中,并且它不会立即执行,只有当满足一定条件后该函数才可以执行。

回调地狱:在回调函数中嵌套回调函数。是实现代码顺序执行的一种方法。

		特点:代码可读性差,调试、维护难度大。

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

Promise对象的本质:是一个异步对象,保存了异步操作的消息,可以看做是异步消息的容器。引入一个回调,避免更多的回调。

	Promise有3个状态:pending[待定] 初始状态
       				  resloved[实现] 操作成功
    			      rejected[被否决] 操作失败

执行过程:

  • resloved函数:返回回调成功的消息,返回值会被.then的回调参数接收
  • rejected函数:返回回调失败的消息,返回值会被.catch的回调参数接收
  • 当Promise对象的状态发生改变时就会触发.then的回调函数
let promise = new Promise((resolve, reject) => {
    setTimeout(() => {
        if (true) {
            resolve({name: '张三'})
        }else {
            reject('失败了') 
        } 
    }, 2000);
});
promise.then(result => console.log(result); // {name: '张三'})
       .catch(error => console.log(error); // 失败了)

异步函数

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

const fn = async () => {};
async function fn () {}

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返回结果

以上是关于Node-异步编程的主要内容,如果未能解决你的问题,请参考以下文章

同步和异步编程有啥区别(在node.js中)

Node.js 同步与异步编程

Node.js之异步编程

Node.js&Promise的新理解&记一次异步编程的错误尝试

Node的异步编程

Node异步