Promise 工具箱:手写实现静态方法的完全指南

news2024/12/25 23:58:14

前言

 📫 大家好,我是南木元元,热爱技术和分享,欢迎大家交流,一起学习进步!

 🍅 个人主页:南木元元


Promise有很多静态方法,本文就来分享下如何实现这些静态方法。

目录

静态方法

实现Promise.resolve和Promise.reject

实现Promise.all

实现Promise.allSettled

实现Promise.race

实现Promise.any

实现finally实例方法

结语


静态方法

静态方法是指直接定义在类上的方法,而不是定义在类实例上的方法。它们可以通过类本身调用,而不需要创建类的实例。静态方法通常用于用于在类级别上操作数据和提供一些实用功能,而不需要实例化对象。比如,可以在 Math 类中定义一些数学计算相关的静态方法。

在上文中,我们已经实现了Promsie中最重要的resolve、reject和then方法,但上文实现的代码是不支持直接使用MyPromise.resolve和MyPromise.reject这种形式的。我们可以在Promise 类上通过static关键字直接定义静态方法,来允许对多个 Promise 对象进行操作或直接创建新的 Promise,而无需实例化一个新的 Promise 对象。

实现Promise.resolve和Promise.reject

MDN上对Promise.resolve()和Promise.reject()的定义:

1.Promise.resolve() 静态方法将给定的值转换为一个 Promise。如果该值本身就是一个 Promise,那么该 Promise 将被返回;如果该值是一个thenable对象,Promise.resolve() 将调用其then() 方法及其两个回调函数;否则,返回的 Promise 将会以该值兑现。

2.Promise.reject() 静态方法返回一个已拒绝(rejected)的Promise  对象,拒绝原因为给定的参数。

class MyPromise {

    ...

    // resolve 静态方法,返回一个以给定值解析后的Promise对象
    static resolve (param) {
        // 1.传参为Promise,直接返回
        if (param instanceof MyPromise) return param;
        
        // 2.直接返回以该值为成功状态的promise对象
        return new MyPromise(resolve =>  {
            resolve(param);
        });
    }

    // reject 静态方法,返回一个带有拒绝原因(拒绝原因为给定参数)的Promise对象
    static reject (reason) {
        return new MyPromise((resolve, reject) => {
            reject(reason);
        });
    }
}

测试:

MyPromise.resolve().then(() => {
    console.log(0);//0
    return MyPromise.resolve(4);
}).then((res) => {
    console.log(res)//4
})

实现Promise.all

all方法用于将多个Promise实例包装成一个新的Promise实例,只有当所有的Promise实例都成功时,新的Promise实例才会成功。

static all(promises) {
    return new Promise(function(resolve, reject) {
        //传入参数为一个空的可迭代对象,直接resolve
        if (promises.length === 0) {
            resolve([]);
        } else {
            const res = [];
            let count = 0;
            for (let i = 0; i < promises.length; i++) {
                //为什么不直接promise[i].then, 因为promise[i]可能不是一个promise, 也可能是普通值
                Promise.resolve(promises[i]).then((data) => {
                    res[i] = data;
                    count++;
                    if (count === promises.length) {
                        resolve(res);//如果所有Promise都成功,则返回成功结果数组
                    }
                }).catch((err) => {
                    reject(err);//如果有一个Promise失败,则返回这个失败结果
                });
            }
        }
    })
}

测试:

const promise1 = MyPromise.resolve(3);
const promise2 = 42;
const promise3 = new MyPromise((resolve, reject) => {
    setTimeout(resolve, 100, "foo");
});

MyPromise.all([promise1, promise2, promise3]).then((values) => {
    console.log(values); //[3, 42, "foo"]
});

实现Promise.allSettled

Promise.allSettled跟Promise.all类似, 唯一的不同在于, 其不会进行短路, 也就是说当Promise全部处理完成后我们可以拿到每个Promise的状态,而不管其是否处理成功。

当有多个彼此不依赖的异步任务成功完成时,或者您总是想知道每个Promise的结果时,通常使用它。如果任务相互依赖,或者如果你想立即拒绝其中任何任务Promise.all()方法更合适。

static allSettled(promises) {
    return new Promise(function(resolve, reject) {
        //传入参数为一个空的可迭代对象,直接resolve
        if (promises.length === 0) {
            resolve([]);
        } else {
            const res = [];
            let count = 0;
            for (let i = 0; i < promises.length; i++) {
                //为什么不直接promise[i].then, 因为promise[i]可能不是一个promise, 也可能是普通值
                Promise.resolve(promises[i]).then((data) => {
                    res[i] = {status: 'fulfilled', value: data};
                    count++;
                    if (count === promises.length) {
                        resolve(res);//如果所有Promise都成功,则返回成功结果数组
                    }
                }).catch((err) => {
                    //失败的时候,不直接返回,而是也把当前状态保存到数组中,等执行完毕时一起返回该数组
                    res[i] = {status: 'rejected', value: err};
                    count++;
                    if (count === promises.length) {
                        resolve(res);
                    }
                });
            }
        }
    })
}

测试:

MyPromise.allSettled([
    MyPromise.resolve(33),
    new MyPromise((resolve) => setTimeout(() => resolve(66), 0)),
    99,
    MyPromise.reject(new Error("an error"))
]).then((values) => {
    console.log(values);
});

实现Promise.race

race 的实现:只要有一个 promise 执行完,直接 resolve 并停止执行。

static race(promises) {
    return new Promise(function(resolve, reject) {
        //传入参数为一个空的可迭代对象,直接resolve
        if (promises.length === 0) {
            resolve([]);
        } else {
            for (let i = 0; i < promises.length; i++) {
                //为什么不直接promise[i].then, 因为promise[i]可能不是一个promise, 也可能是普通值
                Promise.resolve(promises[i]).then((data) => {
                    resolve(data); //返回最快的结果
                }).catch((err) => {
                    reject(err); //返回最快的结果
                });
            }
        }
    })
}

测试:

const promise1 = new MyPromise((resolve, reject) => {
    setTimeout(resolve, 500, 'one');
});
const promise2 = new MyPromise((resolve, reject) => {
    setTimeout(resolve, 100, 'two');
});
MyPromise.race([promise1, promise2]).then((value) => {
    console.log(value);	//two
});

 

实现Promise.any

Promise.any() 静态方法会在任意一个传入的 Promise 成功时,返回该成功的结果。如果所有传入的 Promise 都被拒绝(即失败),它会以AggregateError的形式被拒绝,其中包含所有被拒绝的原因。

static any(promises) {
    return new Promise(function (resolve, reject) {
        //传入参数为一个空的可迭代对象,直接resolve
        if (promises.length === 0) {
            resolve([]);
        } else {
            let count = 0;
            for (let i = 0; i < promises.length; i++) {
                //为什么不直接promise[i].then, 因为promise[i]可能不是一个promise, 也可能是普通值
                Promise.resolve(promises[i])
                .then((data) => {
                    resolve(data); //有一个Promise成功,就返回那个结果
                })
                .catch((err) => {
                    count++;
                    if (count === promises.length) {
                    //当所有输入Promise都被拒绝时,会以一个包含拒绝原因数组的AggregateError拒绝,AggregateError对象代表了包装了多个错误对象的单个错误对象
                    reject(new AggregateError('All promises were rejected')); //所有Promise都失败,就报错
                    }
                });
            }
        }
    });
}

实现finally实例方法

无论Promise是成功还是失败,都会调用finally方法,执行 finally 中传入的函数,并且将值原封不动的往下传,以保证可以继续链式调用。

MyPromise.prototype.finally = function (callback) {
    // 调用then方法,传入两个相同的处理函数,返回一个新的 Promise 对象(保证链式调用)
    return this.then(
        value => {
            // 创建一个新的Promise实例,确保异步执行callback
            return MyPromise.resolve(callback()).then(() => value);
        },
        reason => {
            // 创建一个新的Promise实例,确保异步执行callback
            return MyPromise.resolve(callback()).then(() => { throw reason; });
        }
    );
}

结语

🔥如果此文对你有帮助的话,欢迎💗关注、👍点赞、⭐收藏、✍️评论,支持一下博主~ 

 

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

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

相关文章

Markdown语法与Latex公式汇总

1 基本语法 1.1 标题 语法如下&#xff1a; 效果如下&#xff1a; 1.2 字体样式 语法效果普通正文字体普通正文字体*倾斜字体*倾斜字体**加粗字体**加粗字体***倾斜加粗字体***倾斜字体~~划线字体~~倾斜字体 1.3 分割线 语法如下&#xff1a; 效果如下&#xff1a; …

【C++11及其特性】C++类型转换

C类型转换目录 一.C语言的强制类型转换二.static_cast1.父类子类之间指针或引用的转换2.基本数据类型的转换3.空指针转换其他类型指针4.其他类型指针转换为空指针5.static_cast特点6.完整代码 三.reinterpret_cast1.数值与指针之间的转换2.不同类型指针和引用之间的转换3.reint…

【网络安全】重置邮件逻辑漏洞导致账户接管

未经许可&#xff0c;不得转载。 文章目录 正文 正文 目标&#xff1a;xxx.com 点击重置密码&#xff0c;系统会发送一封链接至邮箱。响应如下&#xff1a; 从上图中可以看到&#xff0c;validationSession对象中有一个sessionID 而收到的链接中的token和sessionID的值是一样…

总结之Coze 是一站式 AI Bot 开发平台——使用coze(一)

Coze 是什么&#xff1f; Coze 是新一代一站式 AI Bot 开发平台。无论你是否有编程基础&#xff0c;都可以在 Coze 平台上快速搭建基于 AI 模型的各类问答 Bot&#xff0c;从解决简单的问答到处理复杂逻辑的对话。并且&#xff0c;你可以将搭建的 Bot 发布到各类社交平台和通讯…

[Leetcode 51][Hard]-n皇后问题-回溯

目录 一、题目描述 二、整体思路 三、代码 一、题目描述 原题地址 二、整体思路 这种可以算是组合问题的变种&#xff0c;在回溯函数中我们要保存当前已放置皇后的所有位置&#xff0c;同时递归调用时要进行寻找下一个皇后的放置位置。那么我们可以逐行遍历棋盘并作为递归调…

STM32学习记录-10-2-SPI通信(硬件)

1 SPI外设简介 STM32内部集成了硬件SPI收发电路,可以由硬件自动执行时钟生成、数据收发等功能,减轻CPU的负担 可配置8位/16位数据帧、高位先行/低位先行 时钟频率: fPCLK / (2, 4, 8, 16, 32, 64, 128, 256) 支持多主机模型、主或从操作 可精简为半双工/单工通信 支持…

python源码 PBOCMaster MAC的计算函数及计算过程 2des

注意最后一步要用整个key加密 计算过程&#xff1a; MAC&#xff1a; PBOC-MAC DES算法 密钥 长度16(0x10)字节 57 75 20 4D 69 61 6F 6A 75 6E 40 47 26 44 43 11 初始向量 长度8(0x08)字节 00 00 00 00 00 00 00 00 数据 长度74(0x4A)字节 43 48 45 4E 48 41 4F 2D 50 43 7…

如何成为一个飞控算法工程师?

兄弟&#xff0c;这个问题问得好&#xff0c;但也别想着靠看几本书就能一步登天。飞控算法这玩意儿&#xff0c;真要干好了&#xff0c;不是简简单单几个公式几个库就能搞定的。你本科电子专业有点基础&#xff0c;玩过四轴飞行器也算是入门了&#xff0c;但要搞真算法&#xf…

做克隆虚拟机的basic

新建一台虚拟机&#xff08;之前写的有这一步&#xff09; 虚拟机里操作 vi /etc/hostname 改称basic (可改可不改) vi /etc/sysconfig/network-scripts/ifcfg-ens33 TYPEEthernet PROXY_METHODnone BROWSER_ONLYno BOOTPROTOstatic DEFROUTEyes IPV4_FAILURE_FATALno IP…

★ 算法OJ题 ★ 力扣 LCR179 - 和为 s 的两个数字

Ciallo&#xff5e;(∠・ω< )⌒☆ ~ 今天&#xff0c;小诗歌剧将和大家一起做一道双指针算法题--和为 s 的两个数字~ 目录 一 题目 二 算法解析 三 编写算法 一 题目 LCR 179. 查找总价格为目标值的两个商品 - 力扣&#xff08;LeetCode&#xff09; 二 算法解析 …

静态工厂模式(简单工厂模式)与动态工厂模式(工厂方法模式)

1. 简单工厂模式 核心是定义一个创建对象的接口&#xff0c;将对象的创建和本身的业务逻辑分离&#xff0c;降低系统的耦合度&#xff0c;使得两个修改起来相对容易些&#xff0c;当以后实现改变时&#xff0c;只需要修改工厂类即可。该模式对对象创建管理方式最为简单&#x…

list的使用与迭代器的模拟实现

前面学习了string&#xff0c;vector类的使用及模拟&#xff0c;但是它们有一个统一的特点就是底层的内存是连续的&#xff0c;因此迭代器的实现也很简单。现在我们开始学习list类的使用&#xff0c;模拟实现&#xff0c;来看看这个底层内存不是连续的有什么特别之处&#xff0…

2024 SEKAI-CTF(nolibc speedpwn life_simulator_2)

文章目录 nolibcexp speedpwnexp life_simulator_2委托构造函数委托构造函数的语法解释 std:remove和std:erase代码解释原理内存管理注意事项 思路1. 背景2. 示例代码3. 解释 vector插入逻辑1. 函数参数2. 本地变量3. 逻辑分析4. 扩容逻辑5. 直接插入逻辑6. 返回结果 exp nolib…

集成电路学习:什么是FPGA现场可编程门阵列

一、FPGA&#xff1a;现场可编程门阵列 FPGA&#xff0c;全称Field Programmable Gate Array&#xff0c;即现场可编程门阵列&#xff0c;是一种超大规模可编程逻辑器件。它由可编程逻辑资源、可编程互连资源和可编程输入输出资源组成&#xff0c;主要用于实现以状态机为主要特…

【计算机组成原理】七、输入/输出系统:2.I/O接口、I/O控制方式

I/O接口、I/O控制方式 2. I/O接口 文章目录 I/O接口、I/O控制方式2. I/O接口2.1 I/O接口的作用2.2 结构2.3 工作原理2.4 I/O端口2.5 分类 3. I/O控制方式3.1程序查询方式3.2程序中断方式3.2.1中断系统3.2.2工作流程3.2.3多重中断与中断屏蔽技术3.2.4程序中断方式 3.3DMA控制方…

Excel技巧(二)

函数 SUMIFS函数 用于计算其满足多个条件的全部参数的总量 语法&#xff1a;SUMIFS(sum_range, criteria_range1, criteria1, [criteria_range2, criteria2], ...) COUNTIFS函数 计算多个区域中满足给定条件的单元格的个数 语法&#xff1a;countifs(criteria_range1,crit…

【Python报错已解决】`ModuleNotFoundError: No module named ‘graphviz‘`

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 引言&#xff1a; 在开发过程中&#xff0c;你是否遇到过尝试导入graphviz模块时遇到了ModuleNotFoundError: No module named …

突发:Runway删库跑路,备受瞩目的Stable Diffusion v1.5不见了!

Runway AI, Inc.创立于2018年&#xff0c;总部位于美国纽约州New York&#xff0c;Runway 是一家应用人工智能研究公司。Runway在谷歌领投的D轮融资中募集到约一亿美元。Runway不仅是投资界的新星&#xff0c;其产品Runway ML参与制作的《瞬息全宇宙》更是斩获了奥斯卡最佳女主…

Mysql基础练习题 1084.销售分析 (力扣)

编写解决方案&#xff0c;报告 2019年春季 才售出的产品。即 仅 在 2019-01-01 &#xff08;含&#xff09;至 2019-03-31 &#xff08;含&#xff09;之间出售的商品 题目链接&#xff1a; https://leetcode.cn/problems/sales-analysis-iii/description/ 建表插入数据&…

【超音速 专利 CN116109587A】一种复杂环境下密封钉焊缝质量检测方法

申请号CN202310066309.X公开号&#xff08;公开&#xff09;CN116109587A申请日2023.01.12申请人&#xff08;公开&#xff09;超音速人工智能科技股份有限公司(833753)发明人&#xff08;公开&#xff09;张俊峰(总); 陈炯标 原文摘要 本发明公开了一种复杂环境下密封钉焊缝…