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 ...
随机推荐
- 信道估计(channel estimation)图解——从SISO到MIMO原理介绍
1. 引言 在所有通信中,信号都会通过一个介质(称为信道),并且信号会失真,或者在信号通过信道时会向信号中添加各种噪声.正确解码接收到的信号而没有太多错误的方法是从接收到的信号中消除信道施加的失真和噪 ...
- 实验五shell脚本编程
项目 内容 这个作业属于哪个课程 <班级课程的主页链接> 这个作业的要求在哪里 作业要求链接地址 学号-姓名 17043220-万文文 作业学习目标 1)了解shell脚本的概念及使用.2 ...
- 深浅拷贝 set集合
数据类型的补充 编码转换 # s1 = '中国' # b1 = s1.encode('utf-8') # # print(b1)-------------->b'\xe4\xb8\xad\xe5 ...
- 怒肝俩月,新鲜出炉史上最有趣的Java小白手册,第一版,每个 Java 初学者都应该收藏
这么说吧,在我眼里,Java 就是最流行的编程语言,没有之一(PHP 往一边站).不仅岗位多,容易找到工作,关键是薪资水平也到位,不学 Java 亏得慌,对吧? 那可能零基础学编程的小伙伴就会头疼了, ...
- Serval and Parenthesis Sequence【思维】
Serval and Parenthesis Sequence 题目链接(点击) Serval soon said goodbye to Japari kindergarten, and began ...
- 装cnpm
npm install -g cnpm --registry=https://registry.npm.taobao.org 然后配置环境变量
- 我要穿越,干翻 “烂语言” JavaScript!
更多精彩文章,尽在码农翻身 我是一个线程 TCP/IP之大明邮差 一个故事讲完Https CPU 阿甘 Javascript: 一个屌丝的逆袭 微服务把我坑了 如何降低程序员的工资? 程序员,你得选准 ...
- 这一次搞懂Spring代理创建及AOP链式调用过程
文章目录 前言 正文 基本概念 代理对象的创建 小结 AOP链式调用 AOP扩展知识 一.自定义全局拦截器Interceptor 二.循环依赖三级缓存存在的必要性 三.如何在Bean创建之前提前创建代 ...
- Python 为什么推荐蛇形命名法?
关于变量的命名,这又是一个容易引发程序员论战的话题.如何命名才能更具有可读性.易写性与明义性呢?众说纷纭. 本期"Python为什么"栏目,我们将聚焦于变量命名中的连接方式,来切入 ...
- linux kernel update
linux内核升级 最近HW行动,报出来的linux系统内核漏洞,环境中全部是2.6.32-431.el6.x86_64的主机,需要全部升级到754版本,这也是第一次进行内核升级操作. 先找了一台和生 ...