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的意思就是承诺// 我们定完外卖,饭不会立即到我们手中// 这时候我们和商家就要达成一个承诺 ...
随机推荐
- [CF9D]How Many Trees?_动态规划_树形dp_ntt
How many trees? 题目链接:https://www.codeforces.com/contest/9/problem/D 数据范围:略. 题解: 水题. $f_{i,j}$表示$i$个节 ...
- TypeScript 命名空间
随着代码的不断增加,我们需要有组织的组合代码.TypeScript在1.x版本中提供了命名空间的方式进行代码组织,这也是TypeScript官方代码的组织方式.同时,TypeScript还实现了Jav ...
- MySQL基础操作(一)
MySQL操作 一.创建数据库 # utf CREATE DATABASE 数据库名称 DEFAULT CHARSET utf8 COLLATE utf8_general_ci; # gbk CREA ...
- java当中JDBC当中请给出一个sql server的stored procedure例子
3.sql server的stored procedure例子: import java.sql.*;public class StoredProc0 {public static void main ...
- JDBC连接池的testQuery/validationQuery设置
在<Tomcat中使用Connector/J连接MySQL的超时问题>帖子中想要增加对连接池中连接的测试/验证,防止数据库认为连接已死而Web应用服务器认为连接还有效的问题,Mysql文档 ...
- NodeJS的环境搭建+传统ELmentui+vue开发
ElementUI简介 我们学习VUE,知道它的核心思想式组件和数据驱动,但是每一个组件都需要自己编写模板,样式,添加事件,数据等是非常麻烦的, 所以饿了吗推出了基于VUE2.0的组件库,它的名称叫做 ...
- Hibernate配置流程
操作数据库必须要设置数据库的连接属性: driver_class,url,username,password(hibernate.cfg.xml) 2. 编写对象跟表之间的映射关系(类名.hb ...
- Security Access Control Strategy && Method And Technology Research - 安全访问控制策略及其方法技术研究
1. 访问控制基本概念 访问控制是网络安全防范和客户端安全防御的重要基础策略,它的主要任务是保证资源不被非法使用.保证网络/客户端安全最重要的核心策略之一. 访问控制包括 入网访问控制 网络权限控制 ...
- Python之数据处理-2
一.数据处理其实是一个很麻烦的事情. 在一个样本中存在特征数据(比如:人(身高.体重.出生年月.年龄.职业.收入...))当数据的特征太多或者特征权重小或者特征部分满足的时候. 这个时候就要进行数据的 ...
- .net通过网络路径下载文件至本地
获取网络文件,通过流保存文件,由于上一版存在数据丢失情况,稍微调整了以下. //网络路径文件 string pathUrl = "http://localhost:805/春风吹.mp3&q ...