Promise入门到精通(初级篇)-附代码详细讲解
Promise入门到精通(初级篇)-附代码详细讲解
Promise,中文翻译为承诺,约定,契约
,从字面意思来看,这应该是类似某种协议,规定了什么事件发生的条件和触发方法。
Promise的诞生和一个词有关,就是异步
什么是
异步???
首先javascript是运行在浏览器端的语言,必须依赖javascript引擎来解析并执行代码,js引擎是
单线程
,也就是一个任务接着一个任务来执行程序,这种单线程很容易因为一个任务发生延迟,造成整体的耗时变长,为了解决这个问题,所以就有了异步
这个概念。 异步就是当系统执行一个事件的时候,不会等待该事件结束,而是会去继续执行其他事件,当这个异步事件有了响应结果之后,系统会在空闲的时候继续执行该事件。
简单来说就是js引擎会首先判断该事件是同步任务还是异步任务,如果是同步任务,将该事件压入宏任务队列中排队等待执行,如果是异步事件,则进入微任务队列中等待宏任务队列处于空闲状态时,再将微任务队列中的事件移入宏任务队列执行。这样我们就可以把不确定执行时间的一些事件用异步来执行。提高了程序运行的效率,
Promise的核心概念
Promise中的核心概念是状态,状态转换就是Promise执行异步事件的时机
在Promise中有存在3种状态,分别对应的是:
1、等待(padding)
2、承诺实现(fulfilled)
3、承诺失效(reject)
Promise初始状态只能为等待的padding状态,在适当的时机,我们可以选择改变padding的状态到fulfilled或者reject。
️ Promise中的状态是不可逆转的,且仅允许改变一次,所以无法从fulfilled或reject状态再次切换到其他状态。当初始的padding改变为fulfilled或reject后,该Promise就相当于完成了它的使命,后续的异步处理就会交由一个then( )的方法来实现。
Promise的基本构成
在ES6语法中,Promise是一个构造函数
,使用时需要用new
关键词来创建实例对象。Promise构造函数中自带excutor
执行器,excutor执行器中有2个JavaScript中默认的函数参数resolve,reject
resolve函数的作用是当Promise状态从padding转换到resolve时,可以把Promise中的对象或者变量当成参数传递出来供异步成功时调用,reject函数的作用是当Promise状态从padding转换到reject时候可以把Promise中的对象或者变量,以及系统报错当成参数传递出来供异步失败时调用。
then是Promise原型上的一个方法,Promise.prototype.then()
所以通过构造函数创建的Promise实例对象也会自带then( )方法。then( )方法接受2个函数参数,作为Promise中异步成功和异步失败的2个回调函数。
Promise实例的基本代码结构:
//ES6 箭头函数写法
let promise = new Promise((resolve,reject)=>{
if(/判断条件/){
resolve()//承诺实现
}else{
reject()//承诺实效
}
})
promise.then(res=>{
//处理承诺实现方法
},err=>{
//处理承诺失效方法
})
️️️注意:Promise函数本身不是一个异步函数,在excutor执行器中运行的代码是同步的。执行异步的是then( )方法中的事件
console.log('步骤1');
new Promise((resolve,reject)=>{
console.log('步骤2');
})
console.log('步骤3')
//执行结果
### 步骤1
### 步骤2
### 步骤3
console.log('步骤1');
new Promise((resolve,reject)=>{
console.log('步骤2');
resolve()
}).then(res=>{
console.log('步骤3');
})
console.log('步骤4')
//执行结果
### 步骤1
### 步骤2
### 步骤4
### 步骤3
.catch也是Promise原型上的一个方法,用来接收和处理Promise中的异步失败,乍一看怎么和then( )中第二个函数参数的功能是一样的嘞?没错滴,这2种方法都是用来处理异步失败的回调函数,但它们2个之间还是有一些小小的区别。 then( )中第二个函数参数只能处理当前Promise异步失败的回调,而catch( )可以处理整个Promise链
上发生的异步失败的回调,便于异步失败和系统报错的整体处理。
let promise new Promise ((resolve,reject)=>{
reject('发生错误了')
})
promise.catch(err=>{
console.log(err)
})
//执行结果
### 发生错误了
let promise new Promise ((resolve,reject)=>{
throw ('发生错误了')
})
promise.catch(err=>{
console.log(err)
})
//执行结果
### 发生错误了
️ (推荐在Promise中使用catch来捕获处理异步失败方法和抛出错误)
Promise链式调用
链式调用是Promise中一个特别重要的属性。也是Promise能控制异步操作的关键,那么链式调用是什么?它的原理又是什么呢?
链式调用最重要的作用就是能使异步事件同步化
,将多个异步事件变为同步,
let promise1 = new Promise((resolve,reject)=>{
setTimeout(()=>{
resolve()
},100)
}).then(res=>{
console.log('我成功了');
})
let promise2 = new Promise((resolve,reject)=>{
resolve()
}).then(res=>{
console.log('我也成功了');
})
//执行结果
### 我也成功了
### 我成功了
可以看出,这两个方法执行顺序并不按照代码结构上的顺序来执行,也就是所谓的异步事件,
链式调用主要原理就是Promise.prototype.then方法和Promise.prototype.catch会返回一个新的Prmise对象,所以我们在Prmise后面可以一直使用then方法来处理异步事件,这样每个异步事件都会等上一个异步事件的then( )方法触发后才会执行自身,从而达到同步的效果.
当我们想让2个异步事件也遵循同步来执行就可以用Prmise的链式调用方法来重写代码结构:
let promise1 = new Promise((resolve,reject)=>{
setTimeout(()=>{
resolve()
},100)
}).then(res=>{
console.log('我成功了');
return new Promise((resolve,reject)=>{
resolve()
})
}).then((res)=>{
console.log('我也成功了');
})
//执行结果
### 我成功了
### 我也成功了
Promise链式调用的规则
链式调用是Promise中一个重要的方法,能有效处理同步异步之间的逻辑关系,但是链式调用也有着自己一套使用规则,熟悉掌握它的规则才能更好的在开发中灵活的使用
规则1:Promise对象会默认返回一个新的Promise,当我们不手动进行干预的时候,这个返回的Promise对象状态为fulfilled
let promise = new Promise((resolve,reject)=>{
resolve()
}).then(res=>{
}).then(res=>{
console.log('规则1');
})
//上述的原理等价于下面的写法
Promise入门到精通(初级篇)-附代码详细讲解的更多相关文章
- YoloV4当中的Mosaic数据增强方法(附代码详细讲解)码农的后花园
上一期中讲解了图像分类和目标检测中的数据增强的区别和联系,这期讲解数据增强的进阶版- yolov4中的Mosaic数据增强方法以及CutMix. 前言 Yolov4的mosaic数据增强参考了CutM ...
- Java入门到精通——基础篇之多线程实现简单的PV操作的进程同步
Java入门到精通——基础篇之多线程实现简单的PV操作的进程同步 一.概述 PV操作是对信号量进行的操作. 进程同步是指在并发进程之间存在一种制约关系,一个进程的执行依赖另一个进程的消 ...
- SaltStack 入门到精通第二篇:Salt-master配置文件详解
SaltStack 入门到精通第二篇:Salt-master配置文件详解 转自(coocla):http://blog.coocla.org/301.html 原本想要重新翻译salt-mas ...
- SaltStack入门到精通第一篇:安装SaltStack
SaltStack入门到精通第一篇:安装SaltStack 作者:纳米龙 发布日期:2014-06-09 17:50:36 实际环境的设定: 系统环境: centos6 或centos5 实验机 ...
- Jmeter(三十四) - 从入门到精通进阶篇 - 参数化(详解教程)
1.简介 前边三十多篇文章主要介绍的是Jmeter的一些操作和基础知识,算是一些初级入门的知识点,从这一篇开始我们就来学习Jmeter比较高级的操作和深入的知识点了.今天这一篇主要是讲参数化,其实前边 ...
- Jmeter(三十五) - 从入门到精通进阶篇 - 关联(详解教程)
1.简介 上一篇中介绍了如果想要同时发送多条请求,那么怎样才能让每条数据某些请求参数改变呢.这就用到了jMeter参数化.在实际测试场景中,我们往往还有这样的需求,登录后服务器响应的token作为下次 ...
- Jmeter(三十八) - 从入门到精通进阶篇 - 命令行运行JMeter详解(详解教程)
1.简介 前边一篇文章介绍了如何生成测试报告,细心地小伙伴或者同学们可以看到宏哥启动Jmeter生成测试报告不是在gui页面操作的,而是在gui页面设置好保存以后,用命令行来生成测试报告的.这一篇宏哥 ...
- Jmeter(四十一) - 从入门到精通进阶篇 - Jmeter配置文件的刨根问底 - 下篇(详解教程)
1.简介 为什么宏哥要对Jmeter的配置文件进行一下讲解了,因为有的童鞋或者小伙伴在测试中遇到一些需要修改配置文件的问题不是很清楚也不是很懂,就算修改了也是模模糊糊的.更有甚者觉得那是禁地神圣不可轻 ...
- Jmeter(四十二) - 从入门到精通进阶篇 - Jmeter配置文件的刨根问底 -番外篇(详解教程)
1.简介 为什么宏哥要对Jmeter的配置文件进行一下讲解了,因为有的童鞋或者小伙伴在测试中遇到一些需要修改配置文件的问题不是很清楚也不是很懂,就算修改了也是模模糊糊的.更有甚者觉得那是禁地神圣不可轻 ...
随机推荐
- Python运算符可不只有加减乘除
数学里面的加减乘除,就是运算符,但是 Python 的运算符更多样,更复杂,分为算术运算符.比较运算符.赋值运算符.位运算符.逻辑运算符.成员运算符.身份运算符.为了更直观的看到运算符的使用,本文采用 ...
- 第 4 篇 Scrum 冲刺博客
每天举行会议 会议照片: 昨天已完成的工作与今天计划完成的工作及工作中遇到的困难: 成员姓名 昨天完成工作 今天计划完成的工作 工作中遇到的困难 蔡双浩 实现收藏夹功能 实现重设计的个人界面功能 无 ...
- 剑指offer二刷——数组专题——斐波那契数列
题目描述 大家都知道斐波那契数列,现在要求输入一个整数n,请你输出斐波那契数列的第n项(从0开始,第0项为0,第1项是1). n<=39 我的想法 斐波那契数列定义:F(0)=0,F(1)=1, ...
- Nginx 转发时的一个坑,运维居然让我背锅!!
最近遇到一个 Nginx 转发的坑,一个请求转发到 Tomcat 时发现有几个 http header 始终获取不到,导致线上出现 bug,运维说不是他的问题,这个锅我背了. 新增的几个 header ...
- linux下安装Zookeeper 3.4.14
1.下载Zookeeper 3.4.14(https://zookeeper.apache.org/) wget https://mirror.bit.edu.cn/apache/zookeeper/ ...
- zstd c++ string 压缩&解压
zstd 简介 维基百科定义: Zstandard(或Zstd)是由Facebook的Yann Collet开发的一个无损数据压缩算法.该名称也指其C语言的参考实现.第1版的实现于2016年8月31日 ...
- sqlplus、lsnrctl命令工具不可用(libclntsh.so.11.1)
原因: libclntsh.so.11.1文件丢失了 解决方法: 在其他机器把这个文件拷贝到目标库安装目录底下的lib目录即可
- mysql 8.0忘记root密码
1.修改参数文件添加以下内容 skip-grant-tables 2.关闭数据库 [root@node01 ~]# /etc/init.d/mysqld8 stop Shutting down MyS ...
- 自动化管理平台rundeck的安装方法
简介 RunDeck 是用 Java/Grails 写的开源工具,帮助用户在数据中心或者云环境中自动化各种操作和流程.通过命令行或者web界面,用户可以对任意数量的服务器进行操作,大大降低了对服务器自 ...
- ipad做windows副屏
利用iPad做windows的触摸显示屏 由于ios与windows不兼容,所以利用软件进行 主要的软件有三款:duet display:spacedesk:Splashtop Wired XDisp ...