手写Promise
构造器的实现
const PENDING = 'pending';
const FULFILLED = 'fulfilled';
const REJECTED = 'rejected'class MyPromise{#state = PENDING;#result = undefined;constructor(executor){const resolve = (data) => {this.#changeState(FULFILLED, data);};const reject = (reason) => {this.#changeState(REJECTED, reason);};try {executor(resolve, reject);} catch (err) {reject(err);} }#changeState(state, result){if(this.#state !== PENDING) return;this.#state = state;this.#result = result;}
}
const p = new MyPromise((resolve, reject) =>{resolve(1);
});
- Promise 的状态一旦定下来之后,在后续的所有过程中都不可能再去改变它的状态了
- 如果再执行过程中报错,会执行rejected,但是捕获不到异步错误
Promise-then的回调执行时机
const PENDING = 'pending';
const FULFILLED = 'fulfilled';
const REJECTED = 'rejected'class MyPromise{#state = PENDING;#result = undefined;#handlers = [];constructor(executor){const resolve = (data) => {this.#changeState(FULFILLED, data);};const reject = (reason) => {this.#changeState(REJECTED, reason);};try {executor(resolve, reject);} catch (err) {reject(err);} }#changeState(state, result){if(this.#state !== PENDING) return;this.#state = state;this.#result = result;this.#run();}#run(){if(this.#state === PENDING) return;while(this.#handlers.length){const {onFulfilled, onRejected, resolve, reject} = this.#handlers.shift();if(this.#state === FULFILLED){if(typeof onFulfilled === 'function'){onFulfilled(this.#result)}}else{if(typeof onRejected === 'function'){onRejected(this.#result)}}}}then(onFulfilled, onRejected){return new MyPromise((resolve, reject) =>{this.#handlers.push({onFulfilled,onRejected,resolve,reject,});this.#run();});}
}const p = new MyPromise((resolve, reject) =>{setTimeout(()=>{resolve(123);},1000)
});p.then((res) =>{console.log('promise 成功1', res);}, (err) =>{console.log('promise 失败1', err);}
);
p.then((res) =>{console.log('promise 成功2', res);}, (err) =>{console.log('promise 失败2', err);}
);
p.then((res) =>{console.log('promise 成功3', res);}, (err) =>{console.log('promise 失败3', err);}
);
为了判断什么时候调用,所以用一个handlers数组记录,并用run函数调用
Promise-then 的返回值
const PENDING = 'pending';
const FULFILLED = 'fulfilled';
const REJECTED = 'rejected'class MyPromise{#state = PENDING;#result = undefined;#handlers = [];constructor(executor){const resolve = (data) => {this.#changeState(FULFILLED, data);};const reject = (reason) => {this.#changeState(REJECTED, reason);};try {executor(resolve, reject);} catch (err) {reject(err);} }#changeState(state, result){if(this.#state !== PENDING) return;this.#state = state;this.#result = result;this.#run();}// 判断一个函数是不是Promise(是否满足A+规范)#isPromiseLike(value){return value && typeof value.then === 'function' && (typeof value === 'object' || typeof value === 'function');}// 把一个函数放到微队列#runMicroTask(func){// node nextTickif(process && process.nextTick){process.nextTick(func);}// 浏览器 MutationObserverelse if(typeof MutationObserver === 'function'){const ob = new MutationObserver(fn);const text = document.createTextNode('1');ob.observe(text, { characterData: true});text.data = '2';}else{setTimeout(fn, 0);}}#runOne(callback, resolve, reject){this.#runMicroTask(()=>{if(typeof callback !== 'function'){const settled = this.#state === FULFILLED ? resolve : reject;settled(this.#result);return;}try {const data = callback(this.#result);if(this.#isPromiseLike(data)){data.then(resolve, reject);}else{resolve(data);}} catch (err) {reject(err);}});}#run(){if(this.#state === PENDING) return;while(this.#handlers.length){const {onFulfilled, onRejected, resolve, reject} = this.#handlers.shift();if(this.#state === FULFILLED){this.#runOne(onFulfilled, resolve, reject);}else{this.#runOne(onRejected, resolve, reject);}}}then(onFulfilled, onRejected){return new MyPromise((resolve, reject) =>{this.#handlers.push({onFulfilled,onRejected,resolve,reject,});this.#run();});}
}const p = new MyPromise((resolve, reject) =>{setTimeout(()=>{reject(123);},1000)
});p.then(null, (err) =>{console.log('promise 失败', err);return 456;}
).then((data) =>{console.log("ok",data);
});
这里有三种情况:
- 对应的回调不是函数的情况
- 回调是一个函数的情况
- 返回结果是一个Promise的情况
相关文章:
手写Promise
构造器的实现 const PENDING pending; const FULFILLED fulfilled; const REJECTED rejectedclass MyPromise{#state PENDING;#result undefined;constructor(executor){const resolve (data) > {this.#changeState(FULFILLED, data);};const reject (reason) > …...
深度学习云服务器免费使用教程
#云服务器# #深度学习# #人工智能# #计算机视觉# 本文为各位学习深度学习的入门选手而创建,降低深度学习的入门门槛。 谷歌云服务器Colab: T4GPU。限额,需要科学上网,不能使用终端。 谷歌云服务器地址:欢迎使用 C…...
使用ansible的剧本制作salt-master与salt-minion的安装与启动服务过程
虚拟机版本:Rocky Linux release 8.6 (Green Obsidian) 准备几台虚拟机 ipv4地址主机名192.168.137.13center192.168.137.14sp-1192.168.137.15sp-2192.168.137.16sp-3 一、center主机的配置 1.vim /etc/hosts 127.0.0.1 localhost localhost.localdomain loc…...
数据库sqlite3
用数据库函数完成数据的增删改查 增: 将要存储的信息录入到结构体中,再使用snprintf函数信息结合sqlite3命令语句使用sqlite3_exec函数完成插入。 int do_insert(sqlite3 *ppDb) {Worker Work;printf("输入插入的工号:");scanf("%d&qu…...
开发基础之Python 函数(Basic Python Functions for Development)
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:Linux运维老纪的首页…...
Django_Vue3_ElementUI_Release_001_项目初始化
1. 数据库 1.1 安装 https://blog.csdn.net/rbx508780/article/details/127176754 1.2 创建数据库 1.3 DBeaver可视化数据库 https://dbeaver.io/download/ 2 安装Python(3.9.12) 2.1 下载地址 https://www.python.org/downloads/release/python-3912/ 2.2 设定国内源 pip …...
MySQL之安装与基础知识
目录 一:在centos7上安装MySQL数据库 1.卸载默认存在的环境 2.配置mysql的yum源 3. 安装MySQL 4.登录mysql 5.设置MySQL的配置文件 二:MySQL基础知识 1.什么是数据库 2.主流数据库 3.服务器,数据库,表关系及使用案例 4…...
前端基础 | HTML基础:HTML结构,HTML常见标签
文章目录 HTML1、HTML结构1.1HTML标签1.1.1标签1.1.2标签含义 1.2HTML文件基本结构1.3标签层次结构1.4 快速生成代码框架 2、HTML常见标签2.1注释标签2.2标题标签:h1–h62.3段落标签:p2.4 换行标签:br2.5格式化标签2.6 图片标签:i…...
宏任务和微任务+超全面试真题
概念 微任务和宏任务是在异步编程中经常使用的概念,用于管理任务的执行顺序和优先级。 宏任务:setTimeout, setInterval,I/O 操作和 UI 渲染等。微任务: Promise 回调、async/await等 微任务通常比宏任务具有更高的优先级。 执…...
针对SVM算法初步研究
归纳编程学习的感悟, 记录奋斗路上的点滴, 希望能帮到一样刻苦的你! 如有不足欢迎指正! 共同学习交流! 🌎欢迎各位→点赞 👍 收藏⭐ 留言📝心态决定高度,细节决定成败…...
Java中的`String`不可变性详解
在Java中,String类具有不可变性(immutable),这意味着一旦String对象被创建,它的值将无法更改。所有对字符串的修改操作(如拼接、替换等)实际上都会生成一个新的字符串对象,而不会修改…...
c# SMTP发送邮件
string from ""; string fromAlias "MIS-TC"; string[] to { "" }; string subject "问题提交"; string body sb.ToString(); string ipaddr "smtp.email.qq.com"; int port 25; string credit ""; strin…...
GPU基础 -- 并行化与阿姆达尔定律
并行化与阿姆达尔定律 并行化是将计算任务分割成多个部分,使这些部分能够在多个处理器或核心上同时运行,从而加速任务的完成时间。阿姆达尔定律(Amdahl’s Law)则揭示了并行化所能带来的加速效果的限制。 阿姆达尔定律公式 阿姆…...
Lua热更
Lua 热更 前提 Lua是轻量级,可以解释执行的编程语言、性能好 基本原则 1.场景空 代码控制物体加载释放 2.场景一个 3.节点不手动挂代码 4.AssetsBundle资源管理 5.Lua开发框架 6.调试模式、发布模式 XLua 热更框架 XLua是C#环境下Lua的解决方案 1.Lua虚拟…...
提升汽车行业软件质量:ASPICE培训的关键实践方法
ASPICE(汽车行业软件过程改进和能力确定)培训是一种针对汽车行业软件开发和维护过程的标准化培训。 该培训旨在帮助组织提高其软件开发和维护过程的质量和效率。以下是ASPICE培训的一些最佳实践方法: 1. 理解ASPICE框架:首先&…...
2024 全新智能识别 API 接口震撼登场
近年来,随着人工智能技术的快速发展,智能识别技术逐渐成为了各个领域的热门应用。在这个大背景下,2024 年的全新智能识别 API 接口横空出世,为我们的生活带来了更多的便利。本文将为大家详细介绍这个全新智能识别 API 接口&#x…...
《UniVS: Unified and Universal Video Segmentation with Prompts as Queries》要点提炼
论文来源:https://arxiv.org/abs/2402.18115 《UniVS: Unified and Universal Video Segmentation with Prompts as Queries》是2024CVPR中的一篇关于视频分割的论文, 主要内容: 论文提出了一个名为UniVS的新型统一视频分割架构,…...
计算机毕业设计选题推荐-推拿知识互动平台-Java/Python项目实战
✨作者主页:IT毕设梦工厂✨ 个人简介:曾从事计算机专业培训教学,擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…...
基于SpringBoot+Vue+MySQL的瑜伽馆管理系统
系统展示 用户前台界面 管理员后台界面 系统背景 本系统采用SpringBoot作为后端框架,Vue.js构建前端用户界面,MySQL作为数据库存储系统,实现了瑜伽馆的全面数字化管理。系统涵盖会员管理、课程预约、教练排班、收入统计等功能模块,…...
【MySQL】EXPLAIN(执行计划)关键字是什么?
简介: explain是一个强大的 SQL 命令,用于分析和优化查询性能。通过查看数据库执行计划,我们可以理解查询是如何被处理的,包括表的访问顺序、使用的索引、连接类型等。这对于找到潜在的性能瓶颈非常重要。 目录 一、基本含义 二…...
模态生成器:原理详解与推荐开源项目
把一种或多种输入模态,转换、补全或生成另一种目标模态的模块。例如: 文本 → 图像 图像 → 文本 文本 → 语音 语音 → 文本 图像 文本 → 视频 图像 文本 → 机器人动作 图像 → 深度图 / mask / 结构化检测结果 缺失模态 → 伪模态补全在 sVLM / ML…...
从“会响”到“可靠”:给这个经典12V降5V电路加个二极管和电容,稳定性提升不止一点点
从“会响”到“可靠”:经典12V降5V电路的稳定性优化实战 当你在面包板上搭建好那个经典的稳压管NPN降压电路,看着万用表显示稳定的5V输出时,或许会感到一丝成就感。但当你接上负载,发现电压开始波动,或者在电源反接时闻…...
One API 部署教程(下):使用指南
导读:前面两篇讲了本地和线上部署,现在 One API 已经跑起来了,接下来就是真正的使用环节! 理解核心概念 在开始之前,咱们先搞清楚几个关键概念,不然后面容易晕。 渠道(Channel):就是你的各个 AI 平台的 API Key。比如你有 DeepSeek 的 Key、OpenAI 的 Key、通义千问…...
初次接触Taotoken的新手如何从注册到完成第一次API调用
🚀 告别海外账号与网络限制!稳定直连全球优质大模型,限时半价接入中。 👉 点击领取海量免费额度 初次接触Taotoken的新手如何从注册到完成第一次API调用 对于初次接触大模型API的开发者而言,从注册平台到成功发出第一…...
特征对高效数值算法及在船舶轴系振动计算中的应用【附仿真】
✨ 长期致力于特征值与特征向量、对称三对角矩阵、振动计算、船舶推进轴系、并行计算研究工作,擅长数据搜集与处理、建模仿真、程序编写、仿真设计。 ✅ 专业定制毕设、代码 ✅ 如需沟通交流,点击《获取方式》 (1)分治并行三对角特…...
tRPC全栈类型安全实战
tRPC全栈类型安全实战:告别API类型地狱,TypeScript前后端零成本类型共享 摘要:在全栈TypeScript项目中,前后端类型不同步是最常见的Bug来源之一。tRPC通过编译时类型推导,实现了端到端的类型安全——前端调用后端API就像调用本地函数一样,类型自动推导、错误提前暴露。本…...
智慧树视频自动播放插件:3分钟搞定所有课程学习的终极指南
智慧树视频自动播放插件:3分钟搞定所有课程学习的终极指南 【免费下载链接】zhihuishu 智慧树刷课插件,自动播放下一集、1.5倍速度、无声 项目地址: https://gitcode.com/gh_mirrors/zh/zhihuishu 还在为智慧树平台繁琐的手动操作而烦恼吗&#x…...
Linux包管理核心:yum机制详解与实战配置指南
1. 项目概述:为什么你需要掌握yum?在Linux的世界里,尤其是以Red Hat、CentOS、Fedora为代表的发行版中,yum(Yellowdog Updater, Modified)是每一位系统管理员和开发者绕不开的核心工具。你可以把它想象成一…...
RedisDesktopManager Windows版:5步打造高效Redis数据库管理体验
RedisDesktopManager Windows版:5步打造高效Redis数据库管理体验 【免费下载链接】RedisDesktopManager-Windows RedisDesktopManager Windows版本 项目地址: https://gitcode.com/gh_mirrors/re/RedisDesktopManager-Windows RedisDesktopManager Windows版…...
现代化管理平台架构优化:FastAPI+Vue3+RBAC权限模型的技术实现与性能提升
现代化管理平台架构优化:FastAPIVue3RBAC权限模型的技术实现与性能提升 【免费下载链接】vue-fastapi-admin ⭐️ 基于 FastAPIVue3Naive UI 的现代化轻量管理平台 A modern and lightweight management platform based on FastAPI, Vue3, and Naive UI. 项目地址…...
