Promise内部实现原理
promise内部实现原理:
function $Promise(fn) {
// Promise 的三种状态
this.PENDING = 'pending'
this.RESOLVED = 'resolved'
this.REJECTED = 'rejected'
this.onResolvedCallback = [] // 成功回调队列
this.onRejectedCallback = [] // 失败回调队列
this.status = this.PENDING // 初始状态
// 成功函数处理
const resove = (val) => {
if (val instanceof $Promise) {
return value.then(resolve, reject)
}
this.triggerResolve(val)
}
// 失败函数处理
const reject = (val) => {
this.triggerReject(val)
}
try {
// 初始同步执行
fn(resove, reject)
} catch (err) {
this.triggerReject(err)
}
}
$Promise.prototype = {
then(onResolved, onRejected) {
// then参数不为函数则使用空函数,缺省函数返回当前值是为了 .then().then() 场景调用
onResolved = typeof onResolved === 'function' ? onResolved : function (value) { return value }
onRejected = typeof onRejected === 'function' ? onRejected : function (reason) { return reason }
// Promise的then方法返回的是新的Promise对象,状态不会更改
return new $Promise((resolve, reject) => {
// 成功回调事件处理
const resolveHandle = val => {
let res = onResolved(val)
if (res instanceof $Promise) {
res.then(resolve, reject)
} else {
resolve(res)
}
}
// 失败回调事件处理
const rejectHandle = val => {
const res = onRejected(val)
if (res instanceof $Promise) {
res.then(resolve, reject)
} else {
reject(res)
}
}
// 当状态已经确认则立即执行
if (this.status === this.RESOLVED) {
return resolveHandle(this.value)
}
if (this.status === this.REJECTED) {
return rejectHandle(this.value)
}
// 当当前状态没有确认,则将回调函数放入队列,等待确认后再一次执行
this.onResolvedCallback.push(resolveHandle)
this.onRejectedCallback.push(rejectHandle)
})
},
// 状态确认后,成功回调队列的函数依次执行
triggerResolve(val) {
let _this = this
setTimeout(() => {
_this.value = val
if (_this.status === _this.PENDING) {
_this.status = _this.RESOLVED
_this.onResolvedCallback.forEach(it => {
it(val)
})
}
})
},
// 状态确认后,失败回调队列的函数依次执行
triggerReject(val) {
let _this = this
setTimeout(() => {
_this.value = val
if (_this.status === _this.PENDING) {
_this.status = _this.REJECTED
_this.onRejectedCallback.forEach(it => {
it(val)
})
}
})
},
// 最后捕捉调用链错误
catch(onRejected) {
return this.then(null, onRejected)
},
// finally实现,不管成功还是失败,都执行
finally(callback) {
return this.then((value) => {
return $Promise.resolve(callback()).then(() => value);
}, (err) => {
return $Promise.resolve(callback()).then(() => err);
});
}
}
// 单独调用Promise.resolve方法实现
$Promise.resolve = val => {
return new $Promise((resolve) => {
resolve(val)
})
}
// 单独调用Promise.reject方法实现
$Promise.reject = val => {
return new $Promise((resolve, reject) => {
reject(val)
})
}
// race函数的实现,返回结果最先完成
$Promise.race = values => {
return new $Promise((resolve, reject) => {
let len = values.length
if (!len) return
for (let i = 0; i < len; i++) {
values[i].then(res => {
resolve(res)
}, error => {
reject(error)
})
}
})
}
// all函数实现,如有错误立即返回,没有错误,等待全部完成再返回
$Promise.all = values => {
return new $Promise((resolve, reject) => {
let len = values.length
if (!len) return
let resolves = []
let nums = 0
function processValue(i, val) {
resolves[i] = val
if (++nums === len) {
resolve(resolves)
}
}
for (let i = 0; i < len; i++) {
values[i].then(res => {
processValue(i, res)
}, error => {
reject(error)
})
}
})
}
Promise内部实现原理的更多相关文章
- Mininet的内部实现原理简介
原文发表在我的博客主页,转载请注明出处. 前言 之前模拟仿真网络一直用的是Mininet,包括写了一些关于Mininet安装,和真实网络相连接,Mininet简历拓扑的博客,但是大多数都是局限于具体步 ...
- KVO内部实现原理
KVO的原理: 只要给一个对象注册一个监听, 那么在运行时, 系统就会自动给该对象生成一个子类对象, (格式如:NSKVONotifying_className), 并且重写自动生成的子类对象的被监听 ...
- Angular单页应用&AngularJS内部实现原理
回顾 自定义指令 登录后获取登录信息session 首先在登录验证的时候保存一个user 在学生管理页面中运用ajax调用获取到登录的用户信息 对注销按钮添加点击事件:调用ajax在表现层给user赋 ...
- 8. 理解ZooKeeper的内部工作原理
到目前为止,我们已经讨论了ZooKeeper服务的基础知识,并详细了解了数据模型及其属性. 我们也熟悉了ZooKeeper 监视(watch)的概念,监视就是在ZooKeeper命名空间中的znode ...
- Redis有序集内部实现原理分析(二)
Redis技术交流群481804090 Redis:https://github.com/zwjlpeng/Redis_Deep_Read 本篇博文紧随上篇Redis有序集内部实现原理分析,在这篇博文 ...
- Apache Lucene评分机制的内部工作原理
Apache Lucene评分机制的内部工作原理' 第5章
- Flask源码分析二:路由内部实现原理
前言 Flask是目前为止我最喜欢的一个Python Web框架了,为了更好的掌握其内部实现机制,这两天准备学习下Flask的源码,将由浅入深跟大家分享下,其中Flask版本为1.1.1. 上次了解了 ...
- 4000余字为你讲透Codis内部工作原理
一.引言 Codis是一个分布式 Redis 解决方案,可以管理数量巨大的Redis节点.个推作为专业的第三方推送服务商,多年来专注于为开发者提供高效稳定的消息推送服务.每天通过个推平台下发的消息数量 ...
- 《转》从系统和代码实现角度解析TensorFlow的内部实现原理 | 深度
from https://www.leiphone.com/news/201702/n0uj58iHaNpW9RJG.html?viewType=weixin 摘要 2015年11月9日,Google ...
随机推荐
- 【Jquery】判断宽度跳转
$(window).resize(function(){ var wWidth = screen.width; if( wWidth < 788 ){ window.location.href= ...
- 秒懂系列,超详细Java枚举教程!!!
所有知识体系文章,GitHub已收录,欢迎Star!再次感谢,愿你早日进入大厂! GitHub地址: https://github.com/Ziphtracks/JavaLearningmanual ...
- Vue好书推荐
1.Vue.js实战 从基础知识到ui组件封装和剖析,层层推进,最后两个案例实战.适合零基础入门,学完可就业.(推荐看这本) 交流地址(pdf原件):链接(点击跳转):提取码:7IsG 2.vue.j ...
- 环境篇:CM+CDH6.3.2环境搭建(全网最全)
环境篇:CM+CDH6.3.2环境搭建(全网最全) 一 环境准备 1.1 三台虚拟机准备 Master( 32g内存 + 100g硬盘 + 4cpu + 每个cpu2核) 2台Slave( 12g内存 ...
- 用python复制文件夹
用python复制文件 1. 根据文件夹的名称复制 需要复制的文件夹编号文件中,每一行表示一个编号,如下所示: > cat id.txt 1 2 3 ... > 目标文件的目录结构树如下所 ...
- (二)Java编程基础
目录 一.关键字与保留字 二.标识符与变量 三.基本数据类型的转换 四.运算符 五.分支语句 五.循环语句 六.跳转语句 七.Java从键盘读取输入 一.关键字与保留字 定义:①关键字:Java关键字 ...
- python flask API 返回状态码
@app.route('/dailyupdate', methods = ['POST','GET'])def dailyUpdate(): try: db=MySQLdb.connect(" ...
- cb22a_c++_标准模板库_STL_map_multimap红黑树(数据结构)关联容器
cb22a_c++_标准模板库_STL_map_multimap红黑树(数据结构)关联容器map(映射,key不能重复,一对一对的,value_type(1, "one")),mu ...
- java递归遍历树结构目录
目录实体 private int id; private int pid; private String name; 递归遍历方法 private void getTree(Menu menu ,Li ...
- Java学习笔记4(多线程)
多线程 多个程序块同时运行的现象被称作并发执行.多线程就是指一个应用程序中有多条并发执行的线索,每条线索都被称作一条线程,它们会交替执行,彼此间可以进行通信. 进程:在一个操作系统中,每个独立执行的程 ...