Promise基础
前言:
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基础的更多相关文章
- promise 基础知识
promise 基础知识 proise:1.Promise是异步编程的一种解决方案,它有三种状态,分别是pending-进行中.resolved-已完成.rejected-已失败2.创建实例//met ...
- promise基础和进阶
本文不对Promise的做过深的解析,只对基础的使用方法,然后会记录一些promise的使用技巧,可以巧妙的解决异步的常见问题. 在过去一直理解的是解决了一直异步回调的坑,但是用了npm async之 ...
- Promise 基础学习
Promise 是ES6的特性之一,采用的是 Promise/A++ 规范,它抽象了异步处理的模式,是一个在JavaScript中实现异步执行的对象. 按照字面释意 Promise 具有"承 ...
- promise基础用法
/** * Created by liyinghao on 2016/11/6. */ const fs = require('fs'); /* * 新建一个Promise对象,Promise就是一个 ...
- JavaScript promise基础示例
const { info } = console // cooking function cook() { info('[COOKING] start cooking.') const p = new ...
- Promise编程基础
(一) Promise基础 所谓Promise,简单说就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果.从语法上说,Promise是一个对象,从它可以获取异步操作的消 ...
- 前端基础进阶(十三):透彻掌握Promise的使用,读这篇就够了
Promise的重要性我认为我没有必要多讲,概括起来说就是必须得掌握,而且还要掌握透彻.这篇文章的开头,主要跟大家分析一下,为什么会有Promise出现. 在实际的使用当中,有非常多的应用场景我们不能 ...
- ES6的promise对象应该这样用
ES6修补了一位Js修真者诸多的遗憾. 曾几何时,我这个小白从js非阻塞特性的坑中爬出来,当我经历了一些回调丑陋的写法和优化的尝试之后,我深深觉得js对于多线程阻塞式的开发语言而言,可能有着其太明显的 ...
- 【JavaScript】JavaScript Promise 探微
http://www.html-js.com/article/Promise-translation-JavaScript-Promise-devil-details 原文链接:JavaScript ...
随机推荐
- Struts2利用注解实现action跳转
使用注解来配置Action的最大好处就是可以实现零配置,但是事务都是有利有弊的,使用方便,维护起来就没那么方便了. 要使用注解方式,我们必须添加一个额外包:struts2-convention-plu ...
- android KeyEvent for dot "."
android连接了4x4的物理按键,需要映射".". 在linux驱动层注册了按键KEY_DOT, 写android的app的时候却没有对应的宏KEYCODE_DOT.只有KEY ...
- urllib2修改header
python网络访问的标准模块 urllib与urllib2并不是升级版的关系,具体可见谷歌文章:difference between urllib and urllib2urllib2的官方文档:h ...
- 15个最佳jQuery的翻页书效果的例子
在这里,你会发现15的jQuery的翻页书的插件,提供了良好的页面翻转的经验,并帮助创建类似书本的效果. jQuery的增添了一道亮丽的过渡到实际的页面在一本书或杂志HTML5. 1. BookBlo ...
- JavaScript(三)---- 控制流程语句
常用的控制流程语句有判断语句.分支语句.循环语句.基本用法都和java中的一致,switch有几点特殊. 1.判断语句 格式: if(判断条件){ 符合条件执行的代 ...
- 微信小程序开发入门教程
做任何程序开发要首先找到其官方文档,微信小程序目前还在邀请内测阶段,目前官方放出了部分开发文档,经过笔者一天的查看和尝试,感觉文档并不全面,但是通过这些文档已经能够看出其大概面貌了.闲话不多说,我们先 ...
- spark 文件系统Alluxio(之前名为Tachyon)
http://www.alluxio.org/documentation/v1.0.0/cn/ http://www.winseliu.com/blog/2016/04/15/alluxio-quic ...
- iOS8学习笔记-构建多视图应用程序
严格来说,凡是UIView的子类都可以称为视图,实际上IOS的图形界面就是视图包含视图的方案,绝大多数的控件都是UIView的子类,但是苹果公司定义的视图术语,通常是指具有相应视图控制器的uiview ...
- js之动态加载等待图像地址汇总
Ajax火啊,火到了居然Loading Icons都有很多人专门提供的地步.下面是我同事给我介绍的一些提供Ajax Activity Indicators的网站,共享给大家,以便让我们的Ajax应用具 ...
- JS实现标签页效果(配合css)不同标签下对应不同div
显示页面tab.jsp </ div ></ body > </ html > tab.css ul ,li { margin:0px; padding:0px ...