Promise 深度学习

news/2025/1/12 4:10:25/

文章目录

    • Promise 由来
    • Promise的用法
    • reject的用法
      • finally
    • all的用法
    • race的用法
    • 总结

Promise 由来

我们处理异步函数最普通的方法是这样的,等待上一次请求结束再执行下一步操作:

// 一般以定时器来模拟一次请求
setTimeout(() => {console.log("first");// 处理的内容 或者 下一次请求// TODO
}, 1000);

这是两次的请求,看着是比较简单,这么写好像也没什么影响,但如果TODO后面还有多个请求时:

// 用多个定时器来模拟多个请求
setTimeout(() => {console.log("第一次处理");setTimeout(() => {console.log("第二次处理");setTimeout(() => {console.log("第三次处理");setTimeout(() => {console.log("第四次处理");// TODO}, 4000)}, 3000)}, 2000)
}, 1000);

回调函数中嵌套回调函数,就是回调地狱,这种写法仔细看我们肯定是能看得出来的,但是无限嵌套之后,代码的可读性非常差,日常维护也是很繁琐的事情。为了更好的处理嵌套格式的代码,我们就可以使用Promise。这当然也是ES6(ES2015)中最重要的特性之一,开发中经常使用,面试的时候也经常问。

Promise的用法

我们可以先打印看看,Promise长什么样:

console.dir(Promise);

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-uMw6cXvW-1687913685665)(C:\Users\zy3\Desktop\study\drop-of-water\water\web\Vue2.0\2023\img\打印出的Promise.png)]

显而易见的Promise是一个构造函数,本身带着 all、race、reject、resolve等方法,prototype原型上也带着 catch、then等方法。new Promise:

// 一般Promise new的时候都喜欢用一个函数包一下,这里还是使用定时器代替请求
const p = new Promise((resolve, reject) => {// 异步操作setTimeout(() => {console.log("时间到了");resolve("成功操作")}, 1000)
});
p.then(res => {console.log(res);
});// 执行后的结果为: 
// 时间到了
// 成功操作

Promise的构造函数接收了一个参数,是一个函数,并且函数中传入两个参数 resolve、reject,分别是异步操作执行成功后的回调函数和异步操作执行失败后的回调函数(这么描述并不是正确的,实际上是 resolve将Promise的状态置为fullfiled,reject是将Promise的状态置为rejected,状态操作是不可逆的,Promise一开始的状态是pending初始态,状态改变方式也就两种)。

上面代码,定时器一秒之后输出“时间到了”,并且调用resolve方法。在异步任务执行完之后,再打印“成功操作”,这就是Promise的作用:将原来的回调写法分离出来,在异步操作执行完之后,用链式调用的写法执行回调函数

这里只是最简单的Promise,如果要改造前面写的多个定时器请求,我们可以这样写:

// Promise优势在于,可以在then方法中继续写Promise对象并返回,然后继续调用then来进行回调操作
const p1 = new Promise((resolve, reject) => {setTimeout(() => {console.log("第一次请求");resolve();}, 1000)
})// 链式操作的用法
p1.then(() => {return new Promise((resolve, reject) => {setTimeout(() => {console.log("第二次请求");resolve()}, 2000)})}).then(() => {return new Promise((resolve, reject) => {setTimeout(() => {console.log("第三次请求");resolve();}, 3000)})}).then(() => {return new Promise((resolve, reject) => {setTimeout(() => {console.log("第四次请求");})}, 4000)})// 换一种写法,执行结果一样 将函数定义一下
function p1() {const p = new Promise((resolve, reject) => {setTimeout(() => {console.log("第一次请求");resolve(1);}, 1000)})return p;
}function p2() {const p = new Promise((resolve, reject) => {setTimeout(() => {console.log("第二次请求");resolve(2);}, 1000)})return p;
}function p3() {const p = new Promise((resolve, reject) => {setTimeout(() => {console.log("第三次请求");resolve(3);})})return p;
}// p1只是返回函数,没有调用,需要在p1后面添加一个 ()
p1().then(res => {console.log(res);return p2();}).then(res => {console.log(res);return p3();}).then(res => {console.log(res);})
// 这样就没有回调地狱了

reject的用法

上面的代码都只用了resolve,我们还没用过reject。事实上前面的代码假设都是成功的,还没有失败的情况:

// 一个在定时器中的随机生成[0,10)随机数的函数
function getNumFn() {const p = new Promise((resolve, reject) => {setTimeout(() => {let num = Math.ceil(Math.random() * 10); // 随机生成1-10(不包含)的随机数if (num <= 5) {// 模拟成功时的操作resolve("数字小于5", num);} else {// 模拟失败时的操作reject("数字大于5", num);}}, 1000)});return p;
};getNumFn().then(res => {console.log("resolve", res);}).catch(err => {console.log("reject", err);})

我们可以看出数字小于5的时候会执行then里的内容,数字大于5时会执行catch中的操作,一般resolve对应then,reject对应catch;但其中then是一个卷王,它可以接收两个参数,一个对应resolve的回调,第二个对应的reject的回调(曾经有个面试官问过我then能接收几个参数,啥也不会),上面的调用也可以写成这样:

getNumFn().then(res => {console.log(res);},err => {console.log(err);}
)

这时就简单达到跟catch一样的效果了。但实际上,在执行resolve的回调时,如果抛出异常了/代码报错了,那么就会卡死,这时catch就显得不可缺少了:

// 这里是没有catch,代码会报错,控制台会出现红色的提示 后面又操作也不会继续执行
getNumFn().then(res => {console.log(num); // 未定义的num,会报错console.log(res);},err => {console.log(num); // 未定义的num,会报错console.log(err);}
)// 写上catch之后
getNumFn().then(res => {console.log(num);console.log(res);},err => {console.log(num);console.log(err);}
).catch(err => {console.log("有错误", err);// 后面有操作可继续执行console.log(123); // 会执行
})
// 控制台的错误提示不会显示为红色

catch方法会把错误原因传到err这个参数中,即便有代码也不会报错,与try/catch语句有相同的功能。

finally

finally方法,这个方法不接受任何参数,但是可以在回调函数中访问之前Promise的解决值或拒绝原因,无论 Promise 的状态如何,最终操作的消息都将打印到控制台。

// 简单拿上面获取随机数函数调一下 (很少用到)
getNumFn().finally(() => {console.log("我都会执行");
})

all的用法

all方法,是在所有异步操作都执行完之后才执行回调:

function p1() {return new Promise((resolve, reject) => {setTimeout(() => {console.log("第一次请求");resolve("操作1");}, 1000)})
}function p2() {return new Promise((resolve, reject) => {setTimeout(() => {console.log("第二次请求");resolve("操作2");})}, 500)
}function p3() {return new Promise((resolve, reject) => {setTimeout(() => {console.log("第三次请求");resolve("操作3");}, 100)})
}Promise.all([ p1(), p2(), p3() ]).then(res => {console.log(res);
})// 执行结果为:
// 第二次请求
// 第三次请求
// 第一次请求
// ["操作1", "操作2", "操作3"]

all方法接收一个数组参数,里面的值最终都算返回Promise对象。这样就算异步操作是并行执行的,all也会等他们全部执行完,才会进入then里面,三个异步操作的数据都被all放到一个数组中了。

race的用法

all方法和race方法区别在于:all全部执行完再执行回调,race一个执行完就执行回调

// 函数还是用上面的 p1、p2、p3
Promise.race([ p1(), p2(), p3() ]).then(res => {console.log(res);
})
// 执行结果:
// 第二次请求
// 操作2 // 这是 Promise.race的执行结果
// 第三次请求
// 第一次请求

总结

Promise不止这些,还有async await语法糖。

看一遍,实际用一遍,你就会了;学吧,学到了都是你自己的。


http://www.ppmy.cn/news/616687.html

相关文章

在pycharm上导出Anaconda3的环境配置文件

目录 1.原理&#xff1a; ​2.亲身实践&#xff1a; 1.原理&#xff1a; 要在PyCharm中导出Anaconda3环境的配置文件&#xff0c;可以使用conda命令行工具来完成。请按照以下步骤进行操作&#xff1a; 打开PyCharm&#xff0c;并确保项目使用的是Anaconda3环境。 在PyCha…

.NET 靠开源再“出圈”!

【编者按】从闭源走向开源&#xff0c;.NET 背后都发生了哪些有趣的故事。本文采访了 6 位微软 .NET 团队成员&#xff0c;分享他们在 GitHub 以及建立 .NET 开源项目的经历。 作者 | Richard Lander 译者 | 弯月 出品 | CSDN&#xff08;ID&#xff1a;CSDNnews&#x…

.NET 5.0来喽

转载&#xff1a;https://blog.csdn.net/powertoolsteam/article/details/109614740?biz_id102&utm_termnet5.0&utm_mediumdistribute.pc_search_result.none-task-blog-2~blog~sobaiduweb~default-0-109614740&spm1018.2118.3001.4450 我们很高兴今天.NET5.0正式…

.NET 5.0正式发布,有什么功能特性(翻译)

我们很高兴今天.NET5.0正式发布。这是一个重要的版本—其中也包括了C# 9和F# 5大量新特性和优秀的改进。微软和其他公司的团队已经在生产和性能测试环境中开始使用了。这些团队向我们反馈的结果比较令人满意&#xff0c;它证明了对性能提升及降低Web应用托管成本的机会有积极的…

Announcing .NET 5.0/发布

We’re excited to release .NET 5.0 today and for you to start using it. It’s a major release — including C# 9 and F# 5 — with a broad set of new features and compelling improvements. It’s already in active use by teams at Microsoft and other companies,…

.NET 5.0正式发布,功能特性介绍(翻译)(转载)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、.NET 5.0亮点1.语言2.记录3.可为空性注释的改进4.Windows窗体设计器 二、.NET 5.0目标框架1.WinRT Interop(重大更改)2.Microsoft.Extensions.Logging3.转储…

.NET 5.0正式发布,功能特性介绍(翻译)

本文由葡萄城技术团队翻译并首发 转载请注明出处&#xff1a;葡萄城官网&#xff0c;葡萄城为开发者提供专业的开发工具、解决方案和服务&#xff0c;赋能开发者。 我们很高兴今天.NET5.0正式发布。这是一个重要的版本—其中也包括了C# 9和F# 5大量新特性和优秀的改进。微软和其…

C++ 多线程 thread

thread 是C 11 引入的 使用的时候需要引入头文件 #include<thread> 头文件中主要包含两个内容&#xff1a;std:thread类和std:this_thread命名空间。 函数名 功能 thread() 构造一个线程对象&#xff0c;没有关联任何线程函数&#xff0c;即没有启动任何线程 t…