如果你也对鸿蒙开发感兴趣,加入“Harmony自习室”吧!扫描下方名片,关注公众号,公众号更新更快,同时也有更多学习资料和技术讨论群。

在鸿蒙的开发中,我们时常会遇到promise异步场景,有同学反馈说希望提一下。
异步开发这部分的内容比较多,我不确定这位朋友具体想讨论是哪些方面,那我从两部分来讨论下,希望能提供一些帮助:
1. 基本的开发角度,常用使用方法;
2. 拿一个问题来讨论调用关系。
【第一部分: 基本使用】
先讨论基本的用法,异步开发中,我们一般会遇到三个关键的内容:Promise、async函数、await命令。
1、Promise
Promise可以看做一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果。
从语法上说,Promise 是一个对象,从它可以获取异步操作的消息。Promise 提供统一的 API,将异步操作以同步操作的流程表达出来,避免了层层嵌套的回调函数。
-
promise异步操作有三种状态:进行中,已成功,已失败。只有异步操作才能改变这个状态。
-
promise状态一旦改变,就不会再发生变化,promise对象改变的两种可能,进行中—>已成功,进行中—>已失败
1.1 基本用法
promise对象是一个构造函数,用来生成promise实例,其中接受的参数是resolve和reject两个函数。
👉🏻 resolve的作用:将promise对象的状态由进行中—>已完成。并将异步操作的结果作为参数传递出去
👉🏻 rejected的作用:将promise对象的状态由进行中—>已失败,并将异步失败的原因作为参数传递出去。
举例:
const promise = new Promise(function(resolve, reject) {// ... some codeif (/* 异步操作成功 */){resolve(value);} else {reject(error);}});
1.2 then方法
promise实例生成后,用then方法分别指定resolved状态和rejucted状态的回调函数。
then方法可以接受两个回调函数作为参数,第一个回调函数是当promise对象状态是resolve(已完成)的时候调用,第二个回调函数(可选)是当promise对象状态是reject(已失败)的时候调用。
举例:
function timeout(ms) {return new Promise((resolve, reject) => {setTimeout(resolve, ms, 'done');});}timeout(100).then((value) => {console.log(value);});// 结果是done
then方法返回的是一个新的Promise实例(注意,不是原来那个Promise实例)。因此可以采用链式写法,即then方法后面再调用另一个then方法。
getjsON("/post/1.json")// 第一个then方法.then(function(post) {return getJSON(post.commentURL);})// 第二个then方法.then(function funcA(comments) {console.log("resolved: ", comments);}, function funcB(err){console.log("rejected: ", err);});
上面代码中,第一个then方法指定的回调函数,返回的是另一个Promise对象。这时,第二个then方法指定的回调函数,就会等待这个新的Promise对象状态发生变化。如果变为resolved,就调用funcA,如果状态变为rejected,就调用funcB。
1.3 catch方法
promise对象中,如果异步操作抛出错误,状态就会变为rejected,就会调用catch方法指定的回调函数处理这个错误,另外,then方法指定的回调函数,如果运行中抛出错误也会被catch方法捕获。
promise对象的错误具有“冒泡”性质,会一直向后传,直到被捕获,也就是说,会跳过中间的then函数。
举例:
getJSON('/post/1.json').then(function(post) {return getJSON(post.commentURL);}).then(function(comments) {// some code}).catch(function(error) {// 处理前面三个Promise产生的错误(一个Promise,两个then)});
1.4 finally方法
finally方法用于指定不管promise对象最后状态如何,都会执行的操作。
举例:
getJSON('/post/1.json').then(function(post) {return getJSON(post.commentURL);}).then(function(comments) {// some code}).finally {// 不论前面三个Promise是异常还是正常,都会执行这里的代码};
1.5 promise.all方法
promise.all方法用于将多个promise实例,包装成一个新的promise实例。
比如:const p = Promise.all([p1, p2, p3]);
Promise.all方法,接受的是一个数组作为参数,其中的元素都是promise实例,如果不是,则会自动将参数转变为promie实例。
p的状态是由它的数组里面的元素决定的,分两种状态
👉🏻 只有p1 p2 p3的状态都变成fulfilled(已完成)的状态才会变成fulfilled(已完成),此时p1 p2 p3的返回值组成一个数组,传递给p的回调函数。
👉🏻 只有p1 p2 p3之中,有一个被rejucted(未完成),p的状态就会变成rejected(未完成),此时第一个被reject的实例的返回值,会传递给p的回调函数。
举例:
const p1 = new Promise((resolve, reject) => {resolve('hello');}).then(result => result);const p2 = new Promise((resolve, reject) => {throw new Error('报错了');}).then(result => result);Promise.all([p1, p2]).then(result => console.log(result)).catch(e => console.log(e));// Error: 报错了enter code here
1.6 promise.race方法
Promise.race方法同样是将多个 Promise 实例,包装成一个新的 Promise 实例,与promise.all不同的是,race中有一个实例率先改变状态,p的状态就跟着改变。那个率先改变的 Promise 实例的返回值,就传递给p的回调函数。(可以将race视为all的一个反面场景)
举例:
// p1、p2、p3之中有一个实例率先改变状态,p的状态就跟着改变const p = Promise.race([p1, p2, p3]);
2、async函数
async函数其实它是Generator函数的语法糖。使异步函数、回调函数在语法上看上去更像同步函数。使得异步操作变得更加方便。基本用法介绍如下:
async返回值是一个promise对象(因此可以使用then方法添加回调函数),当函数执行的时候,一旦遇到await就会先返回,等到异步操作完成,再接着执行函数体内后面的内容。
举例:
async function getStockPriceByName(name) {const symbol = await getStockSymbol(name);const stockPrice = await getStockPrice(symbol);return stockPrice;}getStockPriceByName('goog').then(function (result) {console.log(result);});
async 函数内部return语句返回的值,会成为then方法调用函数的参数。
举例:
async function getTitle(url) {let response = await fetch(url);let html = await response.text();return html.match(/<title>([\s\S]+)<\/title>/i)[1];}getTitle('https://lantingshuxu.github.io').then(console.log)
【需要注意:一个async方法内部如果没有使用await命令,那这个方法相当于是一个普通方法】
~~~停留一下~~~
我们再回顾下前文说到的,① async函数在执行的时候,如果遇到await命令就会先返回;② 如果内部没有遇到await命令,这个方法就相当于是普通方法。
结合上面的描述,下面这段代码的打印顺序是什么呢?[认真想一想]
onBtnClick() {console.log('onClick start');this.funcTest();console.log('onClick end');}async funcTest() {console.log('async func start');await this.funcTest2();console.log('async func end')}async funcTest2() {console.log('async func2');}onBtnClick(); // 代码执行
公布答案,打印顺序为:
onClick startasync func startasync func2onClick endasync func end
解释👇🏻:
-
async函数在没有遇到await指令时和普通方法类似,因此,onBtnClick()函数执行时,会紧接着执行funcTest里面的内容。
-
由于函数执行是从右向左,因此 await this.funcTest2();这段代码会先执行funcTest2(),即打印 async func2,然后再遇到await命令。
-
由于async函数在遇到await指令后,会先返回,因此,按照顺序先打印了 onClick end日志。
-
未来await执行完毕后,继续执行后续逻辑,打印了 async func end。
图解如下:

3、await 命令
正常情况下,await命令后面跟着的是一个promise对象,如果不是会自动转化为promise对象,当一个await语句后面的promise变为reject,那么整个函数都会中断执行。【需要注意的是,await命令只能使用在async函数中,普通函数使用await命令将会报错】
举例:
async function f(){return await 123;}f().then(v =>console.log(v)) // 打印 123async function f2() {await Promise.reject('出错了');await Promise.resolve('hello world'); // 不会执行}f2();
如果await 后面的异步操作有错,那么等同于async函数返回的promis对象被reject (上文讲promise对象的时候有提到过,冒泡性质)。可以使用try ....catch代码块防止出错。
举例:
async function f() {try {await new Promise(function (resolve, reject) {throw new Error('出错了');});} catch(e) {}return await('hello world');}
当然,也可以将多个await命令都放在try..catch结构中。
async function main() {try {const val1 = await firstStep();const val2 = await secondStep(val1);const val3 = await thirdStep(val1, val2);console.log('Final: ', val3);}catch (err) {console.error(err);}}
【第二部分: 场景讨论】
问:假设有下面一段代码,日志的打印顺序将是什么?并说明为什么会这么执行。
async function async1() {console.log( 'async1 start' )await async2()console.log( 'async1 end' )}async function async2() {console.log( 'async2' )}console.log( ' start' )setTimeout(() => { console.log( 'setTimeout' )});async1();new Promise(( resolve ) => {console.log( 'promise1' )resolve(1);}).then(() => console.log( 'promise2' ));console.log( ' end' )
在arcTs中,执行的结果将会是如下:

如果前文的基本用法掌握了,基本上我们应该可以得出正确结果。当然,我们还是简单解释下为什么会出现上面的结果。
我们需要总结几点之前的异步结论:
-
async函数如果没有遇到await命令,执行方式与普通函数相同;
-
await xxxFunc()执行时,xxxFunc()会优先于await命令前执行;
-
await命令会让async函数让出当前时间片,后续的指令将在未来拿到时间片后,等到等待的异步函数执行完毕后再执行;
-
promise中的then是在Promise完成之后执行(相当于是await命令),且then会创建一个新的Promise;
另外,setTimeout和普通的Promise异步还有一个区别,setTimeout属于是一个宏任务,而普通的Promise异步属于是微任务,一般情况下执行顺序是:宏任务执行后再执行微任务,然后再执行宏任务。
因此,一般情况下一次函数执行中,Promise异步任务会先于setTimeout执行。
有了上面的结论,我们看上述代码会怎么执行:
-
console.log( ' start' )是同步代码且顺序靠前,因此最先执行;
-
setTimeout(() => { console.log( 'setTimeout' )}); 添加一个宏任务到宏任务队列,执行时机在promise之后。
-
async1方法虽然是async异步方法,但是在没有遇到await之前,依旧当做同步,因此执行console.log( 'async1 start' ) 打印日志,同时async2() 方法调用也发生在await之前(从右向左看),所以也会执行console.log( 'async2' ) 方法,遇到await之后,方法将会让出当前的执行,因此,await之后的逻辑console.log( 'async1 end' ) 将会在下次时间片去执行。
-
Promise中的实现与async方法类似,由于console.log( 'promise1' ) 在resolve()之前(resolve()之后可以视为await命令之后的逻辑),因此会打印 promise1 。
-
由于Promise的then方法相当于是await之后的逻辑,所以, console.log( 'promise2' ) 也会直接让出时间片。
-
console.log( ' end' ) 由于在同步代码中,因此会立即执行。
在主逻辑执行完后,我们知道,还剩下三个异步任务,分别是:
-
async1() 方法中,await之后的逻辑console.log( 'async1 end' )
-
setTimeout(() => { console.log( 'setTimeout' )});
-
Promise中的then方法 .then(() => console.log( 'promise2' ));
知道setTimeout将会把一个宏任务推送到宏任务队列,因此,执行顺序将是在await和Promise微任务之后。
剩下的 async1()方法中的await逻辑和Promise中的then逻辑,将根据实际情况执行(任务顺序不能完全固定)。因此,最终的顺序为:
startasync1 startasync2promise1endasync1 end // async1中的await先执行promise2setTimeout
或者
startasync1 startasync2promise1endpromise2 // promise中的then先执行async1 endsetTimeout
【尾巴】
我无法确定本文讨论的东西是不是留言的朋友真正疑惑的点,希望这篇文章能帮助你和更多的人。








![[自然语言处理]RNN](https://i-blog.csdnimg.cn/direct/9de5107d0c3e41108bca45e6b40de160.png)










