Node.js异步编程

Node.js 异步编程

所以测试代码在gitee上
https://gitee.com/gaohan888/node-js-learning/tree/master/node%E5%BC%82%E6%AD%A5%E7%BC%96%E7%A8%8B

概念

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

for (let i = 0; i < 10000; i++) {
    console.log(i);
}
console.log('同步代码执行');

只有上面一万行数值打印完,才会打印’同步代码执行’

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

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

setTimeout定时器要在2s秒后才执行,js引擎不会卡在定时器这,会先执行同步代码,等同步代码执行完再执行异步代码定时器(在这只需要先记住定时器是异步代码)

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

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

同步:

// 同步
function sum (n1, n2) { 
	return n1 + n2;
} 
const result = sum(10, 20); // result 值为 30

异步:

function getMsg () { 
	setTimeout(function () { 
		return { msg: 'Hello Node.js' }
	}, 2000);
}
const msg = getMsg(); // msg 的值是 undefined

所以异步函数没法用返回值获取值

回调函数

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

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

function getMsg (callback) {
    setTimeout(() => {
        let a = '异步函数结果'
        callback(a)
    }, 2000)
}

getMsg((result) => {
    console.log(result); // 异步函数结果
})

代码执行顺序分析

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

在这里插入图片描述

异步代码执行区的异步函数执行完成,将要执行专属的回调函数时,就会将回调函数放入回调函数队列,等同步代码执行区的代码执行完成后,就把回调函数队列的回调函数加入同步代码执行区。

Node.js中的异步API

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

读文件函数在读取结束后,会将读到的结果以参数的形式传给回调函数,由回调函数来执行。

var server = http.createServer();
server.on('request', (req, res) => {})

在客户端发送了请求后,将请求的内容以参数的形式传给回调函数的形参res

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

需求:按顺序读取A文件,B文件,C文件

const fs = require('fs')

fs.readFile('a.txt', (err, data) => {
    console.log('第一个执行', data);
    fs.readFile('b.txt', (err, data) => {
        console.log('第二个执行', data);
        fs.readFile('c.txt', (err, data) => {
            console.log(data);
        })
    })
})

连续嵌套着的回调函数可读性非常差,也称为回调地狱

Promise

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

promise小案例展示:

let promise = new Promise((resolve, reject) => {
    setTimeout(() => {
        if (true) {
            resolve({name: 'aaaaa'})
        } else {
            reject('失败了')
        }
    }, 2000);
});

promise.then(result => {
    console.log(result);
}).catch( err => {
    console.log(err);
})

Promise解决 按顺序读取A文件,B文件,C文件 的回调地狱问题

function p1 () {
    return new Promise((resolve, reject) => {
        fs.readFile('a.txt', 'utf-8', (err, data) => {
            resolve(data)
        })
    })
}

function p2 () {
    return new Promise((resolve, reject) => {
        fs.readFile('b.txt', 'utf-8', (err, data) => {
            resolve(data)
        })
    })
}

function p3 () {
    return new Promise((resolve, reject) => {
        fs.readFile('c.txt', 'utf-8', (err, data) => {
            resolve(data)
        })
    })
}

p1().then(r1 => {
    console.log(r1);
    return p2()
}).then(r2 => {
    console.log(r2);
    return p3()
}).then(r3 => {
    console.log(r3);
})

异步函数

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

async 关键字

  1. 普通函数定义前加async关键字 普通函数变成异步函数

    const fn = async () => {};
    async function fn() {}
    
  2. 异步函数默认返回promise对象

    async function fn() {}
    console.log(fn()); // Promise { undefined }
    
  3. 在异步函数内部使用return关键字进行结果返回 结果会被包裹的promise对象中 return关键字代替了resolve方法

    async function fn() {
        return 'abc'
    }
    console.log(fn()); // Promise { 'abc' }
    
  4. 在异步函数内部使用throw关键字抛出程序异常

    async function fn() {
        throw '发生了错误';
        return 'abc'
    }
    console.log(fn());
    

在这里插入图片描述

  1. 调用异步函数再链式调用then方法获取异步函数执行结果

    async function fn() {
        return 'abc'
    }
    // console.log(fn());
    fn().then(res => {
        console.log(res);
    })
    
  2. 调用异步函数再链式调用catch方法获取异步函数执行的错误信息

    async function fn() {
        throw '发生了错误';
        return 'abc'
    }
    // console.log(fn());
    fn().then(res => {
        console.log(res);
    }).catch(err => {
        console.log(err);
    })
    

await关键字

  1. await关键字只能出现在异步函数中

  2. await promise await后面只能写promise对象 写其他类型的API是不不可以的

  3. await关键字可是暂停异步函数向下执行 直到promise返回结果

async function fn1() {
    return 'fn1'
}

async function fn2() {
    return 'fn2'
}

async function fn3() {
    return 'fn3'
}


async function run () {
    let r1 = await fn1();
    let r2 = await fn2();
    let r3 = await fn3();
    console.log(r1);
    console.log(r2);
    console.log(r3);
}

run()

用 async 和 await 解决 按顺序读取A文件,B文件,C文件 的回调地狱问题:

const fs = require('fs')
// fs.readFile() 这个方法不返回Promise对象
// promisify 可以对异步Api进行包装,让包装的方法返回Promise对象以支持 async await 语法
const promisify = require('util').promisify;
const readFile = promisify(fs.readFile);

async function run () {
    let r1 = await readFile('a.txt', 'utf-8')
    let r2 = await readFile('b.txt', 'utf-8')
    let r3 = await readFile('c.txt', 'utf-8')
    console.log(r1);
    console.log(r2);
    console.log(r3);
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值