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的实现方式和运行机制的更多相关文章

  1. js运行机制及异步编程(一)

    相信大家在面试的过程中经常遇到查看执行顺序的问题,如setTimeout,promise,async await等等,各种组合,是不是感觉头都要晕掉了,其实这些问题最终还是考察大家对js的运行机制是否 ...

  2. Windows程序内部运行机制 转自http://www.cnblogs.com/zhili/p/WinMain.html

    一.引言 要想熟练掌握Windows应用程序的开发,首先需要理解Windows平台下程序运行的内部机制,然而在.NET平台下,创建一个Windows桌面程序,只需要简单地选择Windows窗体应用程序 ...

  3. JSP基础总结(运行机制、脚本元素、指令元素、动作元素)

    JSP的运行机制: 1.转译阶段:JSP页面转换成Servlet类: 2.请求阶段:Servlet类执行,将相应结果发送至客户端. 流程解释: 1.用户访问某个JSP页面 2.服务器找到相应的JSP页 ...

  4. Spark Streaming架构设计和运行机制总结

    本期内容 : Spark Streaming中的架构设计和运行机制 Spark Streaming深度思考 Spark Streaming的本质就是在RDD基础之上加上Time ,由Time不断的运行 ...

  5. PHP的运行机制与原理(底层) [转]

    说到php的运行机制还要先给大家介绍php的模块,PHP总共有三个模块:内核.Zend引擎.以及扩展层:PHP内核用来处理请求.文件流.错误处理等相关操作:Zend引擎(ZE)用以将源文件转换成机器语 ...

  6. 谈谈java的运行机制

    1.高级语言的运行机制 我们编程都是用的高级语言(写汇编和机器语言的大牛们除外),计算机不能直接理解高级语言,只能理解和运行机器语言,所以必须要把高级语言翻译成机器语言,计算机才能运行高级语言所编写的 ...

  7. (视频) 《快速创建网站》2.1 在Azure上创建网站及网站运行机制

    现在让我们开始一天的建站之旅. 本文是<快速创建网站>系列的第2篇,如果你还没有看过之前的内容,建议你点击以下目录中的章节先阅读其他内容再回到本文. 访问本系列目录,请点击:http:// ...

  8. Chrome扩展开发之二——Chrome扩展中脚本的运行机制和通信方式

    目录: 0.Chrome扩展开发(Gmail附件管理助手)系列之〇——概述 1.Chrome扩展开发之一——Chrome扩展的文件结构 2.Chrome扩展开发之二——Chrome扩展中脚本的运行机制 ...

  9. 简述JavaScript的运行机制

    想要理解JavaScript的运行机制,需要分别深刻理解以下几个点: · JavaScript的单线程机制 · 任务队列(同步任务和异步任务) · 事件和回调函数 · 定时器 · Event Loop ...

随机推荐

  1. JS规则 编程练习 考考大家的数学,计算以下计算公式的结果。然后在浏览器中运行一下,看看结果是否跟你的结果一致。

    编程练习 考考大家的数学,计算以下计算公式的结果.然后在浏览器中运行一下,看看结果是否跟你的结果一致. 任务 第一步: 在  ? 处填写你的答案. 第二步: 填写完成后,运行一下,看看是不是跟你填写的 ...

  2. linux及windows安装maven

    一.linux安装maven 1.wget http://mirror.bit.edu.cn/apache/maven/maven-3/3.6.1/binaries/apache-maven-3.6. ...

  3. Luogu P2717 寒假作业(平衡树)

    P2717 寒假作业 题意 题目背景 \(zzs\)和\(zzy\)正在被寒假作业折磨,然而他们有答案可以抄啊. 题目描述 他们共有\(n\)项寒假作业.\(zzy\)给每项寒假作业都定义了一个疲劳值 ...

  4. jQuery插件编写,

    jQuery插件编写 jQuery插件 最近搞jquery插件的编写这里做下笔记 给jquery扩展的方式很多,看的我眼花缭乱 方式1 $.fun=function(){} 方式2 $.fn.fun= ...

  5. Python全栈开发:socket代码实例

    客户端与服务端交互的基本流程 服务端server #!/usr/bin/env python # -*- coding;utf-8 -*- import socket sk = socket.sock ...

  6. vue 简单留言本

    代码: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8 ...

  7. 前端-form表单与CSS

    目录 form表单 表单属性 label标签 input标签 select 下拉框标签 textarea多行文本 提交 Flask 结合form表单 初探 CSS介绍以及基本选择器 基本选择器 组合选 ...

  8. LUOGU P1505 [国家集训队]旅游 (树链剖分+线段树)

    传送门 解题思路 快被调死的码农题,,,其实就是一个边权下放到点权的线段树+树剖. #include<iostream> #include<cstdio> #include&l ...

  9. 杂项-公司:Facebook

    ylbtech-杂项-公司:Facebook Facebook(脸书)是美国的一个社交网络服务网站 ,创立于2004年2月4日,总部位于美国加利福尼亚州帕拉阿图,2012年3月6日发布Windows版 ...

  10. PAT甲级——A1067 Sort with Swap(0, i)

    Given any permutation of the numbers {0, 1, 2,..., N−1}, it is easy to sort them in increasing order ...