ES6标准-Promise对象

news2024/11/18 6:15:03

目录

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方法,则只需要写一次

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何利用CSS制作导航菜单

1.利用CSS技术&#xff0c;结合链接和列表&#xff0c;设计并实现“山水之间”页面 示例代码 <!DOCTYPE html> <html><head><meta charset"utf-8"><title>山水之间</title><style>.all{width:900px;}.top{width:900px;h…

Github 2024-11-17 php开源项目日报 Top10

根据Github Trendings的统计,今日(2024-11-17统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量PHP项目10JavaScript项目2Nextcloud服务器:安全的数据之家 创建周期:2796 天开发语言:PHP, JavaScript协议类型:GNU Affero General Public…

【工具变量】2024-2025年地级市异质性数据、城市分组异质性数据(老工业、环境保护、人口流入、沿海等)

一、数据范围&#xff1a; &#xff08;1&#xff09;南北方城市 &#xff08;2&#xff09;东中西城市 &#xff08;3&#xff09;七大地理区、八大综合经济区 &#xff08;4&#xff09;城市群&#xff0c;长三角珠三角京津冀等 &#xff08;5&#xff09;长江流域沿岸、黄河…

响应式网页设计--css

CSS&#xff08;Cascading Style Sheets&#xff0c;层叠样式表&#xff09;是用来控制网页的外观和布局的语言。它与 HTML 一起工作&#xff0c;通过样式定义网页元素的显示方式。CSS 可以控制多种视觉效果&#xff0c;如字体、颜色、布局、间距等。 基本语法&#xff1a; C…

定时器的小应用

第一个项目 第一步&#xff0c;RCC开启时钟&#xff0c;这个基本上每个代码都是第一步&#xff0c;不用多想&#xff0c;在这里打开时钟后&#xff0c;定时器的基准时钟和整个外设的工作时钟就都会同时打开了 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);第二步&…

基于Hadoop、hive的数仓搭建实践

文章目录 架构图Hadoop搭建Hive 搭建MySQL搭建官网文档下载配置配置hive环境变量配置日志文件配置hive-site 复制mysql 驱动包删除日志包初始化元数据启动metastore服务使用hive CLI启动hiveServer2访问hiveserver2客户端连接beeline shell连接 Dbeaver连接经验 基于HDFS Hive…

跨平台WPF框架Avalonia教程 一

安装 安装 Avalonia UI 模板​ 开始使用 Avalonia 的最佳方式是使用模板创建一个应用程序。 要安装 Avalonia 模板&#xff0c;请运行以下命令&#xff1a; dotnet new install Avalonia.Templates 备注 对于 .NET 6.0 及更早版本&#xff0c;请将 install 替换为 --inst…

“fc-async”提供了基本的异步处理能力

在开发中,异步处理已经成为提升系统性能和用户体验的常用方式。然而,传统的@Async注解和基础的异步处理工具在面对复杂的任务场景时,存在局限性。这些局限性包括但不限于高并发环境下的稳定性、任务失败后的恢复机制、以及任务的监控和管理。 开源项目“fc-async”提供了基…

RGB与YCbCr转换算法

目录 RGB与YCbCr转换算法RGB与YCbCr色域介绍RGB模型YCbCr色域简介YCbCr的应用YUV 和 YCbCr 的区别 色彩转换公式 RGB 转 YCbCr 实现RGB 转 YCbCr 的 Matlab 实现RGB 转 YCbCr 的 FPGA 实现 YCbCr 转 RGB 实现YCbCr 转 RGB 的 Matlab 实现YCbCr 转 RGB 的 FPGA 实现 RGB与YCbCr转…

游戏引擎学习第八天

视频参考: https://www.bilibili.com/video/BV1ouUPYAErK/ 理解下面的代码 关于虚函数 代码分解 结构体 foo 的定义&#xff1a; struct foo {int32 X;int64 Y;virtual void Bar(int c); };foo 结构体有两个成员变量&#xff1a;X&#xff08;int32 类型&#xff09;和 Y&…

Vue2教程002:Vue指令

文章目录 2、Vue指令2.1 开发者工具2.2 v-html2.3 v-show和v-if2.4 v-else和v-else-if2.5 v-on2.5.1 内联语句2.5.2 methods 2、Vue指令 2.1 开发者工具 通过谷歌应用商店安装&#xff08;需要科学上网&#xff09;通过极简插件安装 2.2 v-html Vue会根据不同的指令&#x…

计算机视觉 ---图像模糊

1、图像模糊的作用&#xff1a; 减少噪声&#xff1a; 在图像获取过程中&#xff0c;例如通过相机拍摄或者传感器采集&#xff0c;可能会受到各种因素的干扰&#xff0c;从而引入噪声。这些噪声在图像上表现为一些孤立的、不符合图像主体内容的像素变化&#xff0c;如椒盐噪声&…

[JAVA]MyBatis框架—如何获取SqlSession对象实现数据交互(基础篇)

假设我们要查询数据库的用户信息&#xff0c;在MyBatis框架中&#xff0c;首先需要通过SqlSessionFactory创建SqlSession&#xff0c;然后才能使用SqlSession获取对应的Mapper接口&#xff0c;进而执行查询操作 在前一章我们学习了如何创建MyBatis的配置文件mybatis.config.xm…

node.js下载安装步骤整理

>> 进入node.js下载页面下载 | Node.js 中文网 >>点击 全部安装包 >>删除网址node后面部分&#xff0c;只保留如图所示部分&#xff0c;回车 >>点击进入v11.0.0/版本 >>点击下载node-v11.0.0-win-x64.zip(电脑是windows 64位操作系统适用) >…

解决IntelliJ IDEA的Plugins无法访问Marketplace去下载插件

勾选Auto-detect proxy setting并填入 https://plugins.jetbrains.com 代理URL&#xff0c;可以先做检查连接&#xff1a;

解决Jenkins使用 Git 参数插件拉取 commit 列表缓慢问题

Jenkins使用 Git 参数插件拉取 commit 列表缓慢问题 项目问题问题描述解决方案具体实现 项目问题 在 Jenkins 中使用 Git 参数插件 进行参数化构建&#xff0c;具有多方面的重要性和好处。这不仅提高了构建的灵活性和透明度&#xff0c;还能大大提升开发和运维效率。以下是使用…

Pytest-Bdd-Playwright 系列教程(9):使用 数据表(DataTable 参数) 来传递参数

Pytest-Bdd-Playwright 系列教程&#xff08;9&#xff09;&#xff1a;使用 数据表&#xff08;DataTable 参数&#xff09; 来传递参数 前言一、什么是 datatable 参数&#xff1f;Gherkin 表格示例 二、datatable 参数的基本使用三、完整代码和运行效果完整的测试代码 前言 …

Windows内核编程准备

Windows内核编程 驱动 本课程涉及的驱动&#xff0c;是利用驱动程序的高级权限&#xff0c;去做一些用户层不能做的工作&#xff0c;严格来说&#xff0c;这些写出来的东西不能算是驱动&#xff0c;因为它不会驱动任何设备 权限 内核程序是运行在R0级的&#xff0c;因此比应…

uni-app快速入门(八)--常用内置组件(上)

uni-app提供了一套基础组件&#xff0c;类似HTML里的标签元素&#xff0c;不推荐在uni-app中使用使用div等HTML标签。在uni-app中&#xff0c;对应<div>的标签是view&#xff0c;对应<span>的是text&#xff0c;对应<a>的是navigator&#xff0c;常用uni-app…

【代码大模型】Compressing Pre-trained Models of Code into 3 MB论文阅读

Compressing Pre-trained Models of Code into 3 MB key word: code PLM, compression, GA算法 论文&#xff1a;https://dl.acm.org/doi/pdf/10.1145/3551349.3556964 代码&#xff1a;https://github.com/soarsmu/Compressor.git 【why】 1.问题描述&#xff1a; code LLM …