ES6标准-Promise对象

目录

Promise对象的含义

Promise对象的特点

Promise对象的缺点

Promise对象的基本用法

Promise对象的简单例子

Promise新建后就会立即执行

Promise对象回调函数的参数

Promise参数不会中断运行

Promise对象的then方法

Promise对象的catch()方法

Promise状态为resolved再抛出错误是无效的

Promise使用catch()的链式写法

Promise的catch()中也有错误

Promise会吃掉错误

Promise的finally()方法

Promise对象的含义

Promise是异步编程的一种解决方案,比传统解决方案(回调函数事件)---更合理和强大

Promise简单说就是一个容器,里面保存着某个未来才会结束的事件结果

Promise对象的特点

  • Promise对象的状态不受外界影响Promise对象代表一个异步操作,有三种状态:pending(进行中)fulfilled(已成功)rejected(已失败),只有异步操作的结果,可以决定当前是哪一种状态,任何其他操作都无法改变这个状态
  • 一旦状态改变,就不会再变,任何时候都可以得到这个结果。Promise对象的状态改变,只有两种可能:“pending变为fulfilled”和“pending变为rejected”。只要这两种情况发生,状态就凝固了,不会再变了,会一直保持这个结果,这时就称为“resolved(已定型)

Promise对象的缺点

  • Promise对象一旦创建,它就会立即执行,无法中途取消
  • 如果不为Promise对象设置回调函数,Promise对象内部抛出的错误,不会传递到外部
  • 当处于pending状态时,无法得知目前进展到哪个阶段(刚刚开始还是即将完成)

Promise对象的基本用法

ES6规定,Promise对象是一个构造函数,用来生成Promise实例

下面代码是一个Promise示例:

const promise = new Promise((resolve,reject) => {
    //...some code
    if(/* 异步操作成功 */){
        resolve(value);
    }
    else {
        reject(error)
    }
});

Promise构造函数接受一个函数作为参数,该函数的两个参数分别是resolvereject

它们是两个函数

  • resolve函数:将Promise对象的状态从“未完成”变为“成功”,(即从pending变为resolved),在异步操作成功时调用,并将异步操作的结果,作为参数传递出去
  • reject函数:将Promise对象的状态从“未完成”变为“失败”(即从pending变为rejected),在异步操作失败时调用,并将异步操作报出的错误,作为参数传递出去

Promise实例生成以后,可以用then方法分别指定resolved状态和rejected状态的回调函数

promise.then((value) => {
    //success
},(error) => {
    //error
})

then方法可以接收两个回调函数作为参数,第一个回调函数是Promise对象的状态变为resolved时调用,第二个回调函数是Promise对象的状态变为rejected时调用

第二个函数是可选的,并不一定要提供,两个函数都接收Promise对象传出的值作为参数

Promise对象的简单例子

function timeout(ms){
    return new Promise((resolve,reject) => {
        setTimeout(resolve,ms,'done');
    })
}
timeout(100).then((value) => {
    console.log("value值是:",value);
})

效果:

Promise新建后就会立即执行

let promise = new Promise((resolve,reject) => {
    console.log("Promise正在进行");
    resolve();
});

promise.then(() => {
    console.log("promise 成功执行完毕")
})

console.log('间隔符')

上面代码中:

promise对象首先被创建,这个时候Promise已经开始执行

随后,我们调用promise.thn()回调函数,这是一个异步操作,它在等待Promise执行完毕

最后,我们打印出一条“间隔符”的消息,用来演示异步操作的滞后性

效果

Promise对象回调函数的参数

Promise对象中,如果调用resolve函数和reject函数,那么它们的参数会被传递给回调函数

reject函数的参数通常是Error对象的实例,表示跑出的错误

resolve函数的参数除了正常的值以外,还可能是另一个Promise实例,如下面代码所示:

const p1 = new Promise((resolve, reject) => {
    //...
})
const p2 = new Promise((resolve, reject) => {
    //...
    resoleve(p1);
})

p2resolve方法将p1作为参数,即一个异步操作的结果是返回另一个异步操作

注意:“这时p1的状态决定了p2的状态,如果p1是pending,那么p2的回调函数就会等待p1的状态改变,如果p1是resolved或者rejected,那么p2的回调函数会立即执行

再来看一个例子

const p1 = new Promise((resolve,reject) => {
    setTimeout(() => {
        reject(new Error('error'))
    },3000);
})
const p2 = new Promise((resolve,reject) => {
    setTimeout(() => {
        resolve(p1)
    },1000)
})

p2.then((res) => {
    console.log(res)
}).catch((err) => {
    console.log(err)
})

效果

  • 上面代码中,p2的状态依赖于p1的状态,因为p2返回p1,所以p2自身的状态无效了
  • 又因为p2是一个显式抛出错误的语句,所以then()回调无效,被catch()回调捕获

Promise参数不会中断运行

new Promise((resolve,reject) => {
    resolve(1);
    console.log(2);
}).then((res) => {
    console.log(res);
})

效果

上面代码中,调用resolve()函数以后,后面的console.log()依然会执行,并且会首先打印出来

作者是因此resolve()函数会在Promise语句执行完最后一条语句后再执行

一般来说,调用resolvereject以后,Promise的使命就完成了,后续操作应该放到then方法里面,而不应该直接卸载resolvereject后面,所以最好在它们前面加上return语句

new Promise((resolve,reject) => {
    return resolve(1);
    console.log("我不会被打印");
})

Promise对象的then方法

Promise实例具有then方法,也就是说,then方法是定义在原型对象Promise.prototype上的

它的作用是为了Promise实例添加状态改变时的回调函数

then方法的第一个参数是resolved状态的回调函数,第二个参数是rejected状态的回调函数

then方法返回的是一个新的Promise实例(不是原先那个Promise实例),因此可以采用链式写法,即then方法后面再调用另一个then方法

new Promise((resolve,reject) => {
    //some code
}).then((res) => {
    console.log(res)
}).then((res1) => {
    console.log(res1)
});
  • 上面代码使用then方法,依次指定了两个回调函数
  • 第一个回调函数完成以后,会将返回结果作为参数,传入第二个回调函数

采用链式的then,可以指定一组按照次序调用的回调函数。

这时,前一个回调函数,有可能返回的还是一个Promise对象,这时后一个回调函数,就会等待Promise对象的状态发生变化,才会被调用

new Promise((resolve,reject) => {
    resolve(1)
}).then((res) => {
    console.log(res)
    return res
}).then((res1) => {
    console.log(res1)
});

效果

注意:“每次调用then()方法时,都会返回一个不一样的Promise对象,新Promise对象的状态取决于then()方法执行结果

Promise对象的catch()方法

Promise.prototype.catch()方法是.then(null,rejection).then(undefined,rejection)的别名,用于指定发生错误时回调函数

new Promise((resolve,reject) => {
    //some code
}).then((res) => {
    console.log(res);
}).catch((err) => {
    console.log(err);   //捕获New promise和.then()中的错误
});

上面代码中,如果Promise状态为resolved会执行then()方法,如果是rejected或者.then()抛出错误,都会被catch()方法捕获

const promise = new Promise((resolve,reject) => {
    throw new Error('test');
});

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

效果

Promise状态为resolved再抛出错误是无效的

const promise = new Promise((resolve,reject) => {
    resolve('success');
    throw new Error('error');
})

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

promise.catch(error => {    
    console.log(error);
});

效果

可以看到,promise对象在变为resolved状态后,再抛出错误,也不会被catch()捕获

注意:“Promise的状态一旦改变,就永久保持该状态,不会再变了

Promise使用catch()的链式写法

const test = () => {
    return new Promise((resolve,reject) => {
        //下面一行会报错,x是未定义的变量
        resolve(x + 2);
    })
}

test().catch((err) => {
    console.log("promise 错误:",err);
}).then((res) => {
    console.log("我是catch()返回Promise对象的then");
})

效果

  • 上面代码运行完catch()方法指定的回调函数,会接着运行后面那个then()方法指定的回调函数,如果没有报错,则会跳过catch()方法
const test = () => {
    return new Promise((resolve,reject) => {
        resolve("没有错误");
    })
}

test().catch((err) => {
    console.log("promise 错误:",err);
}).then((res) => {
    console.log("我是catch()返回Promise对象的then");
})

效果

因为上面代码没有报错,跳过了catch()方法,直接执行后面的then()方法

此时要是then()方法里面报错,就与前面的catch()无关了

Promise的catch()中也有错误

const test = () => {
    return new Promise((resolve,reject) => {
        //下面一行会报错,x是未定义的变量
        resolve(x + 2);
    })
}

test().catch((err) => {
    console.log("我是test()的错误",err);
    resolve(y + 2);
}).catch((err) => {
    console.log("我是catch()的错误",err);
})    

效果

可以看到,在catch()中出现错误,需要后面再添加一个catch()用来捕获第一个catch()出现的错误,如果不再添加一个catch(),那么第一个catch()的错误将不会被捕获

注意:“在catch()和then()中的错误,会传递到代码外部!!!!

const test = () => {
    return new Promise((resolve,reject) => {
        //下面一行会报错,x是未定义的变量
        resolve(x + 2);
    })
}

test().catch((err) => {
    console.log("我是test()的错误",err);
    y + 2;
})
console.log("我是全局的console.log");

效果

Promise会吃掉错误

如果没有使用catch()方法指定错误处理函数,Promise对象抛出的错误不会传递到外部代码,即不会有任何反应

const test = () => {
    return new Promise((resolve,reject) => {
        //下面一行会报错,x是未定义的变量
        resolve(x + 2);
    })
}

console.log("我正常打印,没有反应");

效果

可以看到,此时没有任何错误信息

这表明Promise内部的错误不会影响到Promise外部的代码

Promise的finally()方法

finally()方法用于指定不管Promise对象最后状态如何,都会执行的操作

promise.then(()=>{...}).catch(()=>{...}).finally(()=>{...})

上面代码中,不管promise最后的状态如何,在执行外thencatch指定的回调函数以后,都会执行finally方法指定的回调函数

  • finally方法的回调函数不接受任何参数,这意味着没有办法知道,前面的Promise状态到底是resolved还是rejected

finally本质上是then()方法的特例:

promise
.finally(() => {
  // 语句
});
// 等同于
promise
.then(
  result => {
    // 语句
    return result;
  },
  error => {
    // 语句
    throw error;
  }
);

上面代码中,如果不使用finally方法,同样的语句需要为成功失败两种情况各写一次

有了finally方法,则只需要写一次

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/918548.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【隐私计算】隐私计算的应用场景探索(大模型隐私计算、隐私数据存储计算、Web3、隐私物联网等)

1. 背景分析 隐私计算作为一种实现“原始数据不出域,可用不可见”的数据流通价值的关键技术,经历了2020-2023年的高光时刻,却在2024年骤然走向低谷。从各种渠道了解到一些业内曾经风光无两的隐私计算公司都有不同程度的裁员。几乎一夜之间&am…

【大数据学习 | flume】flume的概述与组件的介绍

1. flume概述 Flume是cloudera(CDH版本的hadoop) 开发的一个分布式、可靠、高可用的海量日志收集系统。它将各个服务器中的数据收集起来并送到指定的地方去,比如说送到HDFS、Hbase,简单来说flume就是收集日志的。 Flume两个版本区别: ​ 1&…

【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法

【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法 目录 文章目录 【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法目录摘要:研究背景:问题与挑战:如何解…

数据库审计工具--Yearning 3.1.9普民的使用指南

1 页面登录 登录地址:18000 (不要勾选LDAP) 2 修改用户密码 3 DML/DDL工单申请及审批 工单申请 根据需要选择【DML/DDL/查询】中的一种进行工单申请 填写工单信息提交SQL检测报错修改sql语句重新进行SQL检测,如检测失败可以进行SQL美化后…

Day44 | 动态规划 :状态机DP 买卖股票的最佳时机IV买卖股票的最佳时机III

Day44 | 动态规划 :状态机DP 买卖股票的最佳时机IV&&买卖股票的最佳时机III&&309.买卖股票的最佳时机含冷冻期 动态规划应该如何学习?-CSDN博客 本次题解参考自灵神的做法,大家也多多支持灵神的题解 买卖股票的最佳时机【…

Windows配置域名映射IP

一、找到 hosts 文件 打开 C:\Windows\System32\drivers\etc 二、添加hosts文件修改、写入权限 右击hosts文件,点击属性 -> 安全 -> Users -> 编辑 -> Users -> 添加修改、写入权限 -> 确定 -> 确定 进入常规,将只读属性关闭 三、…

sapiens推理的安装与使用

文章目录 1、安装1.1 克隆代码库1.2 设置 Sapiens-Lite 的代码路径1.3 创建 Conda 环境并安装必要的依赖1.4 下载模型检查点 2、推理 sapiens,是meta发布的以人为中心的视觉大模型,"sapiens"这个词来源于拉丁语,意为“智慧的”或“…

黑马智数Day10

项目背景说明 后台管理部分使用的技术栈是Vue2,前台可视化部分使用的技术栈是Vue3 前台可视化项目不是独立存在,而是和后台管理项目共享同一个登录页面 微前端的好处 微前端是一种前端架构模式,它将大型单体应用程序分解为小的、松散耦合的…

A3超级计算机虚拟机,为大型语言模型LLM和AIGC提供强大算力支持

热门大语言模型项目地址:www.suanjiayun.com/mirrorDetails?id66ac7d478099315577961758 近几个月来,我们目睹了大型语言模型(LLMs)和生成式人工智能强势闯入我们的视野,显然,这些模型在训练和运行时需要…

乐维网管平台(七):网络稳定与高效的“安全锦囊”

试想一下,你给电脑升级了一个软件,升级完成后发现有BUG,经常无故卡死,这时候想回退或重新安装旧版本…相对地,一家企业的网络管理员,在对公司的核心交换机进行复杂的配置调整时,一个小小的疏忽&…

基于Python的图片信息推荐系统

作者:计算机学姐 开发技术:SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等,“文末源码”。 专栏推荐:前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏:…

没钱买KEGG怎么办?REACTOME开源通路更强大

之前搜集免费生物AI插图时简单提到了通路数据库Reactome(https://reactome.org/), 那些精美的生物插图只能算是该数据库附赠的小礼品,他的主要功能还是作为一个开源的通路数据库,为相关领域的研究者提供直观的可视化生…

spi 回环

///tx 极性0 (sclk信号线空闲时为低电平) /// 相位0 (在sclk信号线第一个跳变沿进行采样) timescale 1ns / 1ps//两个从机 8d01 8d02 module top(input clk ,input rst_n,input [7:0] addr ,input …

Lc70--319.两个数组的交集(二分查找)---Java版

1.题目描述 2.思路 用集合求交集,因为集合里面的元素要满足不重复、无序、唯一。使得集合在去重、查找和集合操作(如交集、并集、差集等)中非常高效和方便。 3.代码实现 class Solution {public int[] intersection(int[] nums1, int[] nu…

项目2:简易随机数生成器 --- 《跟着小王学Python·新手》

项目2:简易随机数生成器 — 《跟着小王学Python新手》 《跟着小王学Python》 是一套精心设计的Python学习教程,适合各个层次的学习者。本教程从基础语法入手,逐步深入到高级应用,以实例驱动的方式,帮助学习者逐步掌握P…

qml绘制折线图

参考链接 qml绘制折线图 在QML(Qt Modeling Language)中绘制折线图可以通过使用Canvas元素或ChartView元素来实现。以下是两种方法的示例: 方法一:使用Canvas元素 Canvas元素允许你在QML中绘制自定义图形。你可以通过JavaScrip…

MODBUS TCP转CANOpen网关

Modbus TCP转CANopen网关 型号:SG-TCP-COE-210 产品用途 本网关可以实现将CANOpen接口设备连接到MODBUS TCP网络中;并且用户不需要了解具体的CANOpen和Modbus TCP 协议即可实现将CANOpen设备挂载到MODBUS TCP接口的 PLC上,并和CANOpen设备…

Spring Cloud Alibaba [Gateway]网关。

1 简介 网关作为流量的入口,常用功能包括路由转发、权限校验、限流控制等。而springcloudgateway 作为SpringCloud 官方推出的第二代网关框架,取代了Zuul网关。 1.1 SpringCloudGateway特点: (1)基于Spring5,支持响应…

delphi fmx android 离线人脸识别

搜遍全网都没有找到delphi android 能用的 离线人脸识别,无需注册什么开发者 有这方面需求的可以用fsdk 这边用的luxand.FSDK8.0 android下的注册号要自己找下 1,用老猫的工具将android 下的sdk,FSDK.java 编译成FSDK.jar 老猫的工具 2,用上面的工具将FSDK.jar 生成de…

小试银河麒麟系统OCR软件

0 前言 今天在国产电脑上办公,需要从一些PDF文件中复制文字内容,但是这些PDF文件是图片转换生成的,不支持文字选择和复制,除了手工输入,我们还可以使用OCR。 1 什么是OCR OCR (Optical Character Recogni…