Promise原理实现
首先先看一下 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)
}
}
详细解释如下:
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)
}
}
function resolve(value) {
if (that.state === PENDING) {
that.state = RESOLVED
that.value = value
that.resolvedCallbacks.map(cb => cb(that.value))//map这里是执行回调函数
}
}
new MyPromise((resolve, reject) => {
//异步代码
}).then(value => {
console.log(value)
})
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这里是执行回调函数
}
}
new MyPromise((resolve, reject) => {
resolve(1)
}).then(value => {
console.log(value)
})
try {
fn(resolve, reject)
} catch (e) {
reject(e)
}
function resolve(value) {
if (that.state === PENDING) {
that.state = RESOLVED
that.value = value
that.resolvedCallbacks.map(cb => cb(that.value))//map这里是执行回调函数
}
}
new MyPromise((resolve, reject) => {
//同步代码
}).then(value => {
console.log(value)
})
MyPromise.prototype.then = function(onFulfilled, onRejected) {
if (that.state === RESOLVED) {
onFulfilled(that.value)
}
}
Promise原理实现的更多相关文章
- Promise原理 && 简单实现
Promise原理 参考https://github.com/chunpu/promise/blob/master/promise.js 个人认为原博的实现有点问题 在next函数的实现上, 会导致无 ...
- promise原理
简介 Promise 对象用于延迟(deferred) 计算和异步(asynchronous )计算.一个Promise对象代表着一个还未完成,但预期将来会完成的操作.Promise 对象是一个返 ...
- Promise 原理
异步:可同时好几件事,互不影响: 同步:按循序一件一件.... 异步好多缺点:.... promise就是解决异步计算的这些缺点的,主要用于: 1.异步计算: 2.可以将异步操作队列化 按期望的顺序 ...
- 这一次,彻底弄懂 Promise 原理
作者声明 本人将迁移至个人公众号「前端Q」及「掘金」平台写文章.博客园的文章将不再及时更新发布.欢迎大家关注公众号「前端Q」及我的掘金主页:https://juejin.im/user/5874526 ...
- Promise原理—一步一步实现一个Promise
promise特点 一个promise的当前状态只能是pending.fulfilled和rejected三种之一.状态改变只能是pending到fulfilled或者pending到rejected ...
- 30分钟,让你彻底明白Promise原理
前言 前一阵子记录了promise的一些常规用法,这篇文章再深入一个层次,来分析分析promise的这种规则机制是如何实现的.ps:本文适合已经对promise的用法有所了解的人阅读,如果对其用法还不 ...
- Promise原理剖析
传统的异步回调编程最大的缺陷是:回调地狱,由于业务逻辑非常复杂,代码串行请求好几层:并行请求以前也要通过引用step.async库实现.现在ES6推出了Promise,通过Promise的链式调用可以 ...
- ES6之promise原理
我在这里介绍了promise的原理: https://juejin.im/post/5cc54877f265da03b8585902 我在这里 仅仅张贴 我自己实现的简易promise——DiProm ...
- 10分钟,让你彻底明白Promise原理
什么是Promise?本代码用定外卖来举例子,让你明白. // 定外卖就是一个Promise,Promist的意思就是承诺// 我们定完外卖,饭不会立即到我们手中// 这时候我们和商家就要达成一个承诺 ...
随机推荐
- 【GStreamer开发】GStreamer基础教程13——播放速度
目标 快进,倒放和慢放是trick模式的共同技巧,它们有一个共同点就是它们都修改了播放的速度.本教程会展示如何来获得这些效果和如何进行逐帧的跳跃.主要内容是: 如何来变换播放的速度,变快或者变慢,前进 ...
- js 判断字符串是否为JSON格式
function isJSON(str) { if (typeof str == 'string') { try { var obj=JSON.parse(str); if(typeof obj == ...
- 开源软件“meld”-替代beyond compare -- & 放在linux命令后面真好用
1, 使用技巧 命令行直接对比文件 meld dir1 dir2 & 亦可以直接打开界面进行类似beyondCompare的操作. { & 放在命令后面表示设置此进程为后台进程 默认情 ...
- 题解 CF1216C 【White Sheet】
虽然也很水,但这道还是比前两道难多了... 题目大意:给你三个位于同一平面直角坐标系的矩形,询问你后两个是否完全覆盖了前一个 首先,最直观的想法应该是,把第一个矩形内部每个整数点检查一下,看看是否位于 ...
- 全面优化MySQL
MySQL性能瓶颈原因 硬件.系统因素 CPU 磁盘I/O 网络性能 操作系统争用 MySQL相关因素 数据库设计 索引.数据类型 应用程序性能 特定请求.短时事务 配置变量 缓冲区.高速缓存.Inn ...
- Java QuickSelect
Java QuickSelect /** * <html> * <body> * <P> Copyright 1994-2018 JasonInternationa ...
- <a>的javascript+jquery编程实例之删除(定位节点与事件绑定)
相关jquery方法 parent(), remove() //上传图片 article_create.js article_edit.js function uploadAttachment() { ...
- 我们为什么要用redis
Redis的5要点: 1.为什么要选择Redis:介绍Redis的使用场景与使用Redis的原因: 2.Redis常用命令总结:包括时间复杂度总结与具体数据类型在Redis内部使用的数据结构: 3.R ...
- requests模块的基本用法
requests 什么是requests模块 python中封装好的一个基于网络请求的模块 作用 用来模拟浏览器发送请求 环境安装 pip install requests 编码流程 指定 url 发 ...
- 用ajax和PHP实现简单的流程管理
首先要先有一个新建流程的页面xinjian.php <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" ...