promise的实现方式和运行机制
promise的规范其实种类很多,我们最常用的是promise/A+
这篇文章会先将一个类似于promise的构造函数怎么写,网上很多教程并没有实现A+规范,只是看起来像而已
然后我们深入探究如何一步一步实现真正的promise/A+规范的要求。
首先我们实现一个简单的promise,这个promise类似于express的路由处理
express的路由处理以next的方式一层一层的进入, 直到不再next。这个很像promise呐,也是比较懒,没有去看源码,但试着实现一个那种异步,会发现其实蛮简单的。
ok,废话少说,先搞一个
需求
1. promise拥有then和catch方法
2. promise出错时调用catch,有两种错误,主动跑错可以用next(err)
3. promise的第一个参数为function, 第一个参数为next,其他参数可以传递到下一步。
然后我们要实现的功能如下
new Promise(next => {
setTimeout(() => {
console.log();
next(null, 2);
}, );
})
.then((next, arg) => {
setTimeout(() => {
console.log(arg);
next();
}, );
})
.then((next, arg) => {
setTimeout(() => {
console.log(arg);
next();
}, );
})
.catch(err => console.error(err));
实现
首先我们实现then和catch方法
我们建立一个回调队列和一个出错回调1
1)then里的东西都会被推进去
2)catch会去替换默认的回调方法
3)next函数会执行resovleCallbacks里的下一个方法
function Promise(execute) {
let resolveCallbacks = [];
let rejectCallbacks = err => console.error(err);
this.then = fn => {
resolveCallbacks.push(fn);
return this;
};
this.catch = fn => {
rejectCallback = fn;
return this;
};
execute(next);
function next() {
const nextCallback = resolveCallbacks.shift();
nextCallback && nextCallback(next);
}
}
为了捕捉错误
try {
execute(next);
} catch (err) {
rejectCallback(err);
}
我们添加如上的错误捕捉机制
假如果主动跑错就用next(err)
没有出错的话就next(null, arg1, arg2)这样传参
所以我们修改next方法
function next(...args) {
if(args[]){
return rejectCallback(args[]);
}else{
const nextCallback = resolveCallbacks.shift();
try {
nextCallback && nextCallback(next, ...args.slice());
} catch (err) {
rejectCallback(err);
}
}
}
至此就实现完成了,很简单吧
game over
Promise/A+
但事实上一个伟大的promise/A+规范的实现可不止这么简单。
然后我们来看看一个promise/A+规范是什么样子的
1. 状态管理pendding fullfilled(resolved) rejected 三种
2. then返回一个新的promise, then的两个参数是function,是上一步的成功和失败的回调
3. 不同的promise可以接起来使用
以上是主要的规范 更加具体的看这里
要想实现promise/A+,首先要搞清楚它的工作原理
按照规范 最麻烦的是then方法的处理,
但是记住两点就Ok
1. 不同的promise的状态不同,所以then方法不可以返回this,而是应该返回一个新的promise
2. then的执行实在resolve完成后,所以在then方法里把onResolved方法修改为then的第一个参数
3. 在promise中then里的函数如果返回值是promise就必须等到promise执行完成才可以到下一步, 对此我们巧妙的运用promise的then,只要把resolve放到then里边就会延迟resolve的发生
return x.then(resolve);
4. 在promise中then里的函数如果返回值不是promise那就直接执行 然后resolve
5. executor应该是慢一点开始,所以用了setTimeout,因为首先需要执行的是then方法,待一切promise队列就绪后才可以开始。
6. 执行then的时候可能已经resolved了,那么这个时候就立即执行,如果还是pending状态的话那么就放到执行完成的回调函数里。
知道以上几点之后 就可以很轻松的看懂代码了
首先是构造函数
function Promise(executor) {
let onResolved = null;
let onRejected = null;
this.status = 'pending';
let self = this;
let resolve = data => {
self.data = data || null;
self.status = 'resolved';
typeof onResolved === 'function' && onResolved(data);
}
let reject = (err) => {
self.status = 'rejected';
typeof onRejected === 'function' && onRejected(data);
}
this.then = onResolvedFn => {
}
executor(resolve, reject);
}
然后是then方法
if (self.status === 'pending') {
console.log('promise is pending, add callback')
return new Promise(resolve => {
onResolved = data => {
let x = onResolvedFn(self.data)
if (x instanceof Promise) {
return x.then(resolve);
} else {
resolve();
}
}
})
}
if (self.status === 'resolved') {
console.log('promise is resolved, execuate')
return new Promise(resolve => {
let x = onResolvedFn(self.data)
if (x instanceof Promise) {
return x.then(resolve);
} else {
resolve();
}
})
}
then方法实现后,然后我们实现一个Promise.resolve方法
Promise.resolve = data => new Promise(resolve => setTimeout(() => resolve(data)));
关于catch和promise的错误处理机制这里就不详细的写了,反正一切都为了简单看懂。
参考文章: https://zhuanlan.zhihu.com/p/21834559
promise的实现方式和运行机制的更多相关文章
- js运行机制及异步编程(一)
相信大家在面试的过程中经常遇到查看执行顺序的问题,如setTimeout,promise,async await等等,各种组合,是不是感觉头都要晕掉了,其实这些问题最终还是考察大家对js的运行机制是否 ...
- Windows程序内部运行机制 转自http://www.cnblogs.com/zhili/p/WinMain.html
一.引言 要想熟练掌握Windows应用程序的开发,首先需要理解Windows平台下程序运行的内部机制,然而在.NET平台下,创建一个Windows桌面程序,只需要简单地选择Windows窗体应用程序 ...
- JSP基础总结(运行机制、脚本元素、指令元素、动作元素)
JSP的运行机制: 1.转译阶段:JSP页面转换成Servlet类: 2.请求阶段:Servlet类执行,将相应结果发送至客户端. 流程解释: 1.用户访问某个JSP页面 2.服务器找到相应的JSP页 ...
- Spark Streaming架构设计和运行机制总结
本期内容 : Spark Streaming中的架构设计和运行机制 Spark Streaming深度思考 Spark Streaming的本质就是在RDD基础之上加上Time ,由Time不断的运行 ...
- PHP的运行机制与原理(底层) [转]
说到php的运行机制还要先给大家介绍php的模块,PHP总共有三个模块:内核.Zend引擎.以及扩展层:PHP内核用来处理请求.文件流.错误处理等相关操作:Zend引擎(ZE)用以将源文件转换成机器语 ...
- 谈谈java的运行机制
1.高级语言的运行机制 我们编程都是用的高级语言(写汇编和机器语言的大牛们除外),计算机不能直接理解高级语言,只能理解和运行机器语言,所以必须要把高级语言翻译成机器语言,计算机才能运行高级语言所编写的 ...
- (视频) 《快速创建网站》2.1 在Azure上创建网站及网站运行机制
现在让我们开始一天的建站之旅. 本文是<快速创建网站>系列的第2篇,如果你还没有看过之前的内容,建议你点击以下目录中的章节先阅读其他内容再回到本文. 访问本系列目录,请点击:http:// ...
- Chrome扩展开发之二——Chrome扩展中脚本的运行机制和通信方式
目录: 0.Chrome扩展开发(Gmail附件管理助手)系列之〇——概述 1.Chrome扩展开发之一——Chrome扩展的文件结构 2.Chrome扩展开发之二——Chrome扩展中脚本的运行机制 ...
- 简述JavaScript的运行机制
想要理解JavaScript的运行机制,需要分别深刻理解以下几个点: · JavaScript的单线程机制 · 任务队列(同步任务和异步任务) · 事件和回调函数 · 定时器 · Event Loop ...
随机推荐
- Python3基础笔记_字符串类型
# 1.Python转义字符 a = "sqwerdf" # 2.Python字符串运算符 ''' + 字符串连接 a + b 输出结果: HelloPython * 重复输出字符 ...
- CentOS7使用firewalld打开关闭防火墙与端口(转)
CentOS7使用firewalld打开关闭防火墙与端口 1.firewalld的基本使用 启动: systemctl start firewalld 关闭: systemctl stop ...
- JS 作用域、原型链
看到一道好题,并附答案 function Foo() { getName = function () { console.log('1'); }; return this; } Foo.getName ...
- 立体声耳机接口3F07
仅探讨三段式 剩余的两根,则是开关,插入耳机后,会断开,可以用于耳机和扬声器的切换.老式收音机大多采用这个结构
- if __name__=='__main__'使用场景,彻底明白
本博中有一篇文章写了 if __name__=='__main__'的作用与原理http://www.cnblogs.com/fennudexiaoniao/p/7458324.html,但是好像似懂 ...
- Deepin折腾手记之安装常用软件
1. 创建快捷方式 在创建快捷图标的文件/usr/share/applications/xx.desktop 编辑内容 [Desktop Entry] Name=VNote X-Deepin-Vend ...
- VRRP概述-转
本文介绍了VRRP的基本原理.特点和应用. VRRP概述 随着Internet的发展,人们对网络的可靠性的要求越来越高.对于局域网用户来说,能够时刻与外部网络保持联系是非常重要的. 通常情况下,内部网 ...
- 查看MySql版本号命令
转自:https://blog.csdn.net/qq_38486203/article/details/80324014 这里介绍四中不同的方法,它们分别运行在不同的环境中,最后对每种方法的优劣以 ...
- shell下时间日期的加减乘除运算
首先我们先来说说什么是shell下的时间戳: 自1970年1月1日(00:00:00 UTC/GMT)以来的秒数.它也被称为Unix时间戳(Unix Timestam.Unix epoch.POSIX ...
- 跟我一起学习webpack输出动态HTML(三)
跟着之前的项目来 我们没打包一次就会生成一个bundile.js,我们要更新最新的代码不希望有缓存,那么这个时候我们就是更改资源的URL, 每当代码发生变化时,相应的hash也会发生变化.这个时候我们 ...