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内部实现原理的更多相关文章

  1. Mininet的内部实现原理简介

    原文发表在我的博客主页,转载请注明出处. 前言 之前模拟仿真网络一直用的是Mininet,包括写了一些关于Mininet安装,和真实网络相连接,Mininet简历拓扑的博客,但是大多数都是局限于具体步 ...

  2. KVO内部实现原理

    KVO的原理: 只要给一个对象注册一个监听, 那么在运行时, 系统就会自动给该对象生成一个子类对象, (格式如:NSKVONotifying_className), 并且重写自动生成的子类对象的被监听 ...

  3. Angular单页应用&AngularJS内部实现原理

    回顾 自定义指令 登录后获取登录信息session 首先在登录验证的时候保存一个user 在学生管理页面中运用ajax调用获取到登录的用户信息 对注销按钮添加点击事件:调用ajax在表现层给user赋 ...

  4. 8. 理解ZooKeeper的内部工作原理

    到目前为止,我们已经讨论了ZooKeeper服务的基础知识,并详细了解了数据模型及其属性. 我们也熟悉了ZooKeeper 监视(watch)的概念,监视就是在ZooKeeper命名空间中的znode ...

  5. Redis有序集内部实现原理分析(二)

    Redis技术交流群481804090 Redis:https://github.com/zwjlpeng/Redis_Deep_Read 本篇博文紧随上篇Redis有序集内部实现原理分析,在这篇博文 ...

  6. Apache Lucene评分机制的内部工作原理

    Apache Lucene评分机制的内部工作原理' 第5章

  7. Flask源码分析二:路由内部实现原理

    前言 Flask是目前为止我最喜欢的一个Python Web框架了,为了更好的掌握其内部实现机制,这两天准备学习下Flask的源码,将由浅入深跟大家分享下,其中Flask版本为1.1.1. 上次了解了 ...

  8. 4000余字为你讲透Codis内部工作原理

    一.引言 Codis是一个分布式 Redis 解决方案,可以管理数量巨大的Redis节点.个推作为专业的第三方推送服务商,多年来专注于为开发者提供高效稳定的消息推送服务.每天通过个推平台下发的消息数量 ...

  9. 《转》从系统和代码实现角度解析TensorFlow的内部实现原理 | 深度

    from https://www.leiphone.com/news/201702/n0uj58iHaNpW9RJG.html?viewType=weixin 摘要 2015年11月9日,Google ...

随机推荐

  1. 循序渐进VUE+Element 前端应用开发(7)--- 介绍一些常规的JS处理函数

    在我们使用VUE+Element 处理界面的时候,往往碰到需要利用JS集合处理的各种方法,如Filter.Map.reduce等方法,也可以设计到一些对象属性赋值等常规的处理或者递归的处理方法,以前对 ...

  2. 什么!你想要封装好的ajax

    ajax作为前端开发领域一个必不可少的内容,也是灵魂所在,今天就ajax的封装给大家做一个分析, 如果没有猜错的话现在基本上用原生去写ajax的意见不多了,这是肯定的 ,为什么这么说,jq的ajax大 ...

  3. 基于ABP做一个简单的系统——实战篇:1.项目准备

    现阶段需要做一个小项目,体量很小,业务功能比较简单,就想到用最熟悉的.net来做,更何况现在.net core已经跨平台,也可以在linux服务器上部署.所以决定用.net core 3.1+mysq ...

  4. Mysql(Mariadb)数据库主从

    Mysql主从复制的实现原理图大致如下: MySQL之间数据复制的基础是以二进制日志文件(binary log file)来实现的,一台MySQL数据库一旦启用二进制日志后,其作为master,它数据 ...

  5. 01.Markdown学习

    Markdown学习 一.标题 在想要设置为标题的文字前面加#来表示(#后面有空格) 一个#是一级标题,二个#是二级标题,以此类推.支持六级标题. 示例: # 这是一级标题 ## 这是二级标题 ### ...

  6. DeDecms远程写入漏洞webshell (dedecms漏洞)

    解释下Apache解析文件的流程: 当Apache检测到一个文件有多个扩展名时,如1.php.bak,会从右向左判断,直到有一个Apache认识的扩展名.如果所有的扩展名Apache都不认识,那么变会 ...

  7. JFinal 源码解析-MVC部分

    首先从请求入口看起,应用初始化时加载web.xml的JFinalFilter,和configClass 从这段配置可以看出jfinal和spring mvc入口类似,通过一个实现Servlet Fil ...

  8. (十三)exec-maven-plugin配置及使用

    原文链接:https://www.cnblogs.com/lianshan/p/7358966.html 背景: 如果你想在项maven生命周期内,运行一段java代码,或者一段独立的程序,或者说我们 ...

  9. MySQL——事务(Transaction)详解

    原文:https://blog.csdn.net/w_linux/article/details/79666086

  10. C++ 进阶 模板和STL

    C++提高编程 本阶段主要针对C++泛型编程和STL技术做详细讲解,探讨C++更深层的使用 1 模板 1.1 模板的概念 模板就是建立通用的模具,大大提高复用性 模板的特点: 模板不可以直接使用,它只 ...