前言:

ES2015将Promise引入语言规范,包括fetch等在内的API也构建在Promise之上。作为让js摆脱“回调地狱”的重要一环和众多框架中的重要基础设施之一,学习如何自己实现一个Promise插件依然是前端工程师晋级之路上很有意义的练习之一。

Promise规范:

在众多Promise规范中,最著名的就是A+规范:

在此规范中,promise被定义为一个带有then方法的object或function,具有pending(可转换为fulfilled或rejected),fulfilled(不可转换,需带有一个value),rejected(不可转换,需带有一个reason)三种状态。

then方法调用时接受成功和失败的回调:promise.then(onFulfilled, onRejected)。onFulfilled在promise被fulfill时调用一次,并传入value;onRejected在promise被reject时调用一次,传入reason。一个promise的then可以被多次调用,因此调用then需要返回promise。

Promise可以被resolve,resolve的传入值也可以是一个promise(thenable的对象)。正确解析promise链,处理其中的resolve、exception、handler的返回值等等是关键所在。

具体文档可见:https://promisesaplus.com/

自己实现Promise:

mmDeferred(https://github.com/RubyLouvre/mmDeferred)是一个优秀的遵循Promise/A+规范的Deffered库,我们就来仿造它看看如何写一个Deffered库。

Deferred函数:

function Deferred() {

var state = "pending", dirty = false // 定义状态state,dirty

function ok(x) { // 定义转换state为fulfilled的接口,返回传入的x

state = "fulfilled"

return x

}

function ng(e) {

state = "rejected" // 定义转换state为rejected的接口,抛出异常

throw e

}

...

}

接下来我们添加Deferred的返回,它将是一个thenable的对象:

var dfd = {

callback: { // 定义resolve与reject的callback

resolve: ok,

reject: ng,

},

dirty: function() {

return dirty

},

state: function() {

return state

},

promise: {

then: function() { // 定义接口then,接受onFulfilled,onRejected作为参数

return _post.apply(null, arguments)

},

_next: null // 每个thenable通过_next完成promise链的构成

}

}

最后我们将给dfd对象绑定resolve与reject对象并返回:

"resolve,reject".replace(/\w+/g, function(method) {

dfd[method] = function() {

var that = this, args = arguments

// promise链上的第一个回调需要异步触发,但剩余链上的回调需要同步触发

if (that.dirty()) {

_fire.call(that, method, args)

} else {

setTimeout(function() {

_fire.call(that, method, args)

}, 0);

}

}

})

return dfd

这里用到了两个闭包内的函数_post与_fire,我们来看看他们各自做了什么:

function _post() {

var index = -1, fns = arguments;

"resolve,reject".replace(/\w+/g, function(method) {

var fn = fns[++index];

if (typeof fn === "function") {

// promise链的头结点,回调需要异步触发

dirty = true

// 修改dfd的resolve与reject函数,绑定then函数传入的回调

dfd.callback[method] = function() {

try {

var value = fn.apply(this, arguments)

state = "fulfilled"

return value

} catch (err) {

state = "rejected"

return err

}

}

}

})

// 形成promise链

var deferred = dfd.promise._next = Deferred(mixin)

return deferred.promise;

}

function _fire(method, array) {

var next = "resolve", value

// 根据状态触发回调

if (this.state() === "pending") {

var fn = this.callback[method]

try {

value = fn.apply(this, array);

} catch (e) {//处理notify的异常

value = e

}

if (this.state() === "rejected") {

next = "reject"

}

array = [value]

}

// promise链依次触发

var nextDeferred = this.promise._next

if (Deferred.isPromise(value)) {

value._next = nextDeferred

} else {

if (nextDeferred) {

_fire.call(nextDeferred, next, array);

}

}

}

还有isPromise 函数:

Deferred.isPromise = function(obj) {

return !!(obj && typeof obj.then === "function");

};

以上便是简化的Deferred库实现,在原库中还有一些优化,比如Deferred可以接受一个对象,以扩展这个对象为thenable,以及在resolve和reject之外还定义了notify(同时增加notify状态,可以继续转化为fulfilled或者rejected状态)以及ensure(无论状态如何转换都会触发),定义了Deferred.all与Deferred.any等等。

Promise基础的更多相关文章

  1. promise 基础知识

    promise 基础知识 proise:1.Promise是异步编程的一种解决方案,它有三种状态,分别是pending-进行中.resolved-已完成.rejected-已失败2.创建实例//met ...

  2. promise基础和进阶

    本文不对Promise的做过深的解析,只对基础的使用方法,然后会记录一些promise的使用技巧,可以巧妙的解决异步的常见问题. 在过去一直理解的是解决了一直异步回调的坑,但是用了npm async之 ...

  3. Promise 基础学习

    Promise 是ES6的特性之一,采用的是 Promise/A++ 规范,它抽象了异步处理的模式,是一个在JavaScript中实现异步执行的对象. 按照字面释意 Promise 具有"承 ...

  4. promise基础用法

    /** * Created by liyinghao on 2016/11/6. */ const fs = require('fs'); /* * 新建一个Promise对象,Promise就是一个 ...

  5. JavaScript promise基础示例

    const { info } = console // cooking function cook() { info('[COOKING] start cooking.') const p = new ...

  6. Promise编程基础

    (一) Promise基础     所谓Promise,简单说就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果.从语法上说,Promise是一个对象,从它可以获取异步操作的消 ...

  7. 前端基础进阶(十三):透彻掌握Promise的使用,读这篇就够了

    Promise的重要性我认为我没有必要多讲,概括起来说就是必须得掌握,而且还要掌握透彻.这篇文章的开头,主要跟大家分析一下,为什么会有Promise出现. 在实际的使用当中,有非常多的应用场景我们不能 ...

  8. ES6的promise对象应该这样用

    ES6修补了一位Js修真者诸多的遗憾. 曾几何时,我这个小白从js非阻塞特性的坑中爬出来,当我经历了一些回调丑陋的写法和优化的尝试之后,我深深觉得js对于多线程阻塞式的开发语言而言,可能有着其太明显的 ...

  9. 【JavaScript】JavaScript Promise 探微

    http://www.html-js.com/article/Promise-translation-JavaScript-Promise-devil-details 原文链接:JavaScript ...

随机推荐

  1. Python之路: socket篇

    (默认)与特定的地址家族相关的协议,如果是  ,则系统就会根据地址格式和套接类别,自动选择一个合适的协议 sk import socketip_port = ()sk = socket.socket( ...

  2. EXCEL读写NPOI--导出功能

    第一步:将NPOI中的一下三个文件复制到项目中

  3. The 2014 ACMICPC Asia Regional Xian Online

    [A]签到题 [B]后缀数组 [C]染色,DP(感觉可出) [D]BFS搜索,有点麻烦 [E]博弈论,Nim博弈 [F]BFS状态搜索 [G]概率DP+状态压缩 [H]异或+构造 [I]矩阵快速幂(队 ...

  4. Python之路【第三篇】:模块

    定义: 包:包含__init__.py文件.模块(也是.py文件) 当包被其它模块调用时,首先会执行该包下的__init__文件 包含有模块,包可以有多级 模块的导入: import from...i ...

  5. hibernate---一对一单向主键关联(不重要)

    比如, husband的id参考wife的id husband.java: package com.bjsxt.hibernate; import javax.persistence.Entity; ...

  6. 检测网站挂马程序(Python)

    系统管理员通常从svn/git中检索代码,部署站点后通常首先会生成该站点所有文件的MD5值,如果上线后网站页面内容被篡改(如挂马)等,可以比对之前生成MD5值快速查找去那些文件被更改,为了使系统管理员 ...

  7. [Unity Shader]Shader分类

    Shader的分类: Shader按管线分类一般分为固定渲染管线与可编程渲染管线 (1)固定渲染管线 ——这是标准的几何&光照(Transforming&Lighting)管线,功能是 ...

  8. WIN32动态链接库设计与使用

    WINDOWS动态链接库技术能很好地实现代码的分模块,综合来说,windows动态链接库分为三种WIN32动态链接库,使用WINDOWS api函数调用设计,贴近底层,体积小,是最初Windows程序 ...

  9. memcache细节解析

    转自:原链接 Memcached内存管理采取预分配.分组管理的方式,分组管理就是划分slab class,按照chunk的大小slab被分为很多种类.   slab Slab是一个内存块,它是memc ...

  10. RFID射频卡超市购物结算系统问题记录--写入卡片时,后台php无法操作数据库

    后台管理人员要给每件商品贴上RF卡作为唯一标识,所以要先给对应的RFID卡中写入响应的信息,我这里为了便于模拟演示只写入商品编号,价格,名称这几个字段,然后要把已经写入的商品上传后台,由后台写入数据库 ...