首先先看一下 promise 的调用方式:

// 实例化 Promise:
new MyPromise((resolve, reject) => {
setTimeout(() => {
resolve(1) //这里相当于给value赋值
}, 0)
}).then(value => {
console.log(value)
})

实现原理如下:

const PENDING = 'pending'  //首先我们创建了三个常量用于表示状态,对于经常使用的一些值都应该通过常量来管理,便于开发及后期维护
const RESOLVED = 'resolved'
const REJECTED = 'rejected' function MyPromise(fn) {
const that = this //在函数体内部首先创建了常量 `that`,因为代码可能会异步执行,用于获取正确的 `this` 对象
that.state = PENDING //一开始 `Promise` 的状态应该是 `pending`
that.value = null //`value` 变量用于保存 `resolve` 或者 `reject` 中传入的值
that.resolvedCallbacks = []
that.rejectedCallbacks = []
/*
`resolvedCallbacks` 和 `rejectedCallbacks` 用于保存 `then` 中的回调,
因为当执行完 `Promise` 时状态可能还是等待中,这时候应该把 `then` 中的回调保存起来用于状态改变时使用
*/
function resolve(value) {
if (that.state === PENDING) {
that.state = RESOLVED
that.value = value
that.resolvedCallbacks.map(cb => cb(that.value))//map这里是执行回调函数
}
} function reject(value) {
if (that.state === PENDING) {
that.state = REJECTED
that.value = value
that.rejectedCallbacks.map(cb => cb(that.value))//map这里是执行回调函数
}
}
/*
* 首先两个函数都得判断当前状态是否为等待中,因为规范规定只有等待态才可以改变状态
* 将当前状态更改为对应状态,并且将传入的值赋值给 `value`
* 遍历回调数组并执行
*/
try {
fn(resolve, reject)
} catch (e) {
reject(e)
} } MyPromise.prototype.then = function(onFulfilled, onRejected) {
const that = this
//首先判断两个参数是否为函数类型,因为这两个参数是可选参数
//当参数不是函数类型时,需要创建一个函数赋值给对应的参数,同时也实现了透传
onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : (v) => { return v }
onRejected =
typeof onRejected === 'function'
? onRejected
: r => {
throw r
}
//接下来就是一系列判断状态的逻辑,当状态不是等待态时,就去执行相对应的函数。
//如果状态是等待态的话,就往回调函数中 `push` 函数,比如如下代码就会进入等待态的逻辑
if (that.state === PENDING) {
that.resolvedCallbacks.push(onFulfilled)//push这里是保存回调函数
that.rejectedCallbacks.push(onRejected)
}
if (that.state === RESOLVED) {
onFulfilled(that.value)
}
if (that.state === REJECTED) {
onRejected(that.value)
}
}

详细解释如下:

定义异步函数 MyPromise,所以执行的函数也是 MyPromise:
首先看 函数执行的方法:
new MyPromise((resolve, reject) => {
setTimeout(() => {
resolve(1)
}, 0)
})
函数的参数是:
(resolve, reject) => {
setTimeout(() => {
resolve(1)
}, 0)
}
对应着
function MyPromise(fn){
try {
fn(resolve, reject) // 在这里执行了传入的参数fn(),并且把规定的
resolve, reject 两个参数传递到 fn 中。
} catch (e) {
reject(e)
}
}
中的 fn,所以会执行这个传入的函数 fn(resolve, reject);
传入的参数是异步的,会在同步代码结束后再去执行对应的 resolve(1)这个函数,
而这个函数已经在 MyPromise 中进行了定义:
function resolve(value) {
if (that.state === PENDING) {
that.state = RESOLVED
that.value = value
that.resolvedCallbacks.map(cb => cb(that.value))//map这里是执行回调函数
}
}
// 也就是会在同步代码之后再执行上面的函数,所以我们继续看 MyPromise 的调用
new MyPromise((resolve, reject) => {
//异步代码
}).then(value => {
console.log(value)
})
接着执行方法: then()。而then 挂在了原型链上:
MyPromise.prototype.then = function(onFulfilled, onRejected) {  }
所以then包含两个参数,分别对对应 onFulfilled, onRejected。 如果没有定义需要做容错处理,
也就是给默认的函数参数: onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : (v) => { return v } 
由于执行同步操作,此时state 还等于PENDING,所以执行:
 
if (that.state === PENDING) {
that.resolvedCallbacks.push(onFulfilled)//push这里是保存回调函数
that.rejectedCallbacks.push(onRejected)
}
也就是把 成功后操作函数和失败函数分别保存到对应的数组中。
setTimeout(() => {
resolve(1)
}, 0)
好了,接下来同步执行结束,然后开始执行异步操作:
function resolve(value) {
if (that.state === PENDING) {
that.state = RESOLVED
that.value = value
that.resolvedCallbacks.map(cb => cb(that.value))//map这里是执行回调函数
}
}
可以看到,先改变状态,再从保存数组中,获取到回调函数,再执行!
至此,在promise中 resolve(1) 告诉了执行回调的时机和参数。
而then规定的是异步之后的回调函数。
然后我们看到还在then函数中规定了其他的状态,解释一下:
如果执行函数中,没有异步处理:
new MyPromise((resolve, reject) => {
resolve(1)
}).then(value => {
console.log(value)
})
也就是在定义中:
try {
fn(resolve, reject)
} catch (e) {
reject(e)
}
直接执行了 fn(),根据传入的 resolve,reject。这里直接执行了 resolve:
function resolve(value) {
if (that.state === PENDING) {
that.state = RESOLVED
that.value = value
that.resolvedCallbacks.map(cb => cb(that.value))//map这里是执行回调函数
}
}
更改了状态,由于还没有执行then函数,保存函数没有数据,所以没有可以执行的回调函数。
接下来程序走到了then
new MyPromise((resolve, reject) => {
//同步代码
}).then(value => {
console.log(value)
})
由于状态已经改变,所以执行:
MyPromise.prototype.then = function(onFulfilled, onRejected) {
if (that.state === RESOLVED) {
onFulfilled(that.value)
}
}
同样也做到了执行then传入的函数。
总结一下,如果 resolve 被放在了异步函数中,then传入的回调函数会先被保存下来,待异步函数执行完毕之后,
在次执行回调函数;
而如若 resolve 被放在了同步函数中,则回调函数数组为空,顺序执行到 then 函数,则会执行该回调函数。

Promise原理实现的更多相关文章

  1. Promise原理 && 简单实现

    Promise原理 参考https://github.com/chunpu/promise/blob/master/promise.js 个人认为原博的实现有点问题 在next函数的实现上, 会导致无 ...

  2. promise原理

      简介 Promise 对象用于延迟(deferred) 计算和异步(asynchronous )计算.一个Promise对象代表着一个还未完成,但预期将来会完成的操作.Promise 对象是一个返 ...

  3. Promise 原理

    异步:可同时好几件事,互不影响: 同步:按循序一件一件.... 异步好多缺点:.... promise就是解决异步计算的这些缺点的,主要用于: 1.异步计算: 2.可以将异步操作队列化  按期望的顺序 ...

  4. 这一次,彻底弄懂 Promise 原理

    作者声明 本人将迁移至个人公众号「前端Q」及「掘金」平台写文章.博客园的文章将不再及时更新发布.欢迎大家关注公众号「前端Q」及我的掘金主页:https://juejin.im/user/5874526 ...

  5. Promise原理—一步一步实现一个Promise

    promise特点 一个promise的当前状态只能是pending.fulfilled和rejected三种之一.状态改变只能是pending到fulfilled或者pending到rejected ...

  6. 30分钟,让你彻底明白Promise原理

    前言 前一阵子记录了promise的一些常规用法,这篇文章再深入一个层次,来分析分析promise的这种规则机制是如何实现的.ps:本文适合已经对promise的用法有所了解的人阅读,如果对其用法还不 ...

  7. Promise原理剖析

    传统的异步回调编程最大的缺陷是:回调地狱,由于业务逻辑非常复杂,代码串行请求好几层:并行请求以前也要通过引用step.async库实现.现在ES6推出了Promise,通过Promise的链式调用可以 ...

  8. ES6之promise原理

    我在这里介绍了promise的原理: https://juejin.im/post/5cc54877f265da03b8585902 我在这里 仅仅张贴 我自己实现的简易promise——DiProm ...

  9. 10分钟,让你彻底明白Promise原理

    什么是Promise?本代码用定外卖来举例子,让你明白. // 定外卖就是一个Promise,Promist的意思就是承诺// 我们定完外卖,饭不会立即到我们手中// 这时候我们和商家就要达成一个承诺 ...

随机推荐

  1. Linux中执行 .sh 的方法

    linux下执行.sh文件的方法 .sh文件就是文本文件,如果要执行,需要使用chmod a+x xxx.sh来给可执行权限. 是bash脚本么 可以用touch test.sh #创建test.sh ...

  2. SSH 连接时间超时

    linux服务端 # vi /etc/ssh/sshd_config ClientAliveInterval 60 ClientAliveCountMax 3 # 注: # ClientAliveIn ...

  3. PHP中NULL和‘'的区别

    PHP中NULL和‘'区别 null的类型是null,""的类型是string 所以是不同东西 <pre>$a=22;unset($a);var_dump($a);&l ...

  4. 洛谷 题解 UVA247 【电话圈 Calling Circles】

    [题意] 如果两个人互相打电话(直接或者间接),则说他们在同一个电话圈里.例如,\(a\)打给\(b\),\(b\)打给\(c\),\(c\)打给\(d\),\(d\)打给\(a\),则这四个人在同一 ...

  5. MySQL的索引有哪些

    一.索引是什么 索引,在MySQL中也叫“键(key)”,是存储引擎用于快速找到记录的一种数据结构.如果把数据库的一张表比作一本书,那索引则是这本书的目录,通过目录,我们能快速找到我们想要的主题所对应 ...

  6. Machine Learning Glossary

    http://people.seas.harvard.edu/~mgelbart/glossary.html more at:http://www.metacademy.org/list

  7. VMware虚拟机(Ubuntu)通过主机代理实现——浏览器+终端访问外网

    环境说明:主机win10 + 虚拟机ubunut16.04 + 主机s-h-a-d-o-w-socks win10 主机相关操作配置1: 按下 Win + R 快捷键,输入 cmd ,然后在命令行中输 ...

  8. maven一些简单常用却容易记混的命令参数-U -e -B

    install 命令完成了项目编译.单元测试.打包功能,同时把打好的可执行jar包(war包或其它形式的包)布署到本地maven仓库,但没有布署到远程Maven私服仓库: deploy 命令完成了项目 ...

  9. 滤波器算法(1)-卡尔曼滤波小车附带题目与MATLAB程序

    1 简介 由卡尔曼这个学者提出的最佳线性滤波器,单纯时域维度即可实现[无需进行频域变换] 2 思路 由上一时刻的最佳估计值XKE_P预测①当前时刻预测值Pxv 与 ②当前时刻的测量值Mxv 进行联立计 ...

  10. Tomcat Lifecycle

    org.apache.catalina.Lifecycle 接口统一管理生命周期,所有生命周期组件都要实现Lifecycle接口. 该接口定义了13个String类型的常量,用于LifecycleEv ...