随便说说Promise
为啥要说 promise ?
因为这是前端必须要掌握的一个知识,吹逼必备
首先说说 Promise 是什么?
Promise 是JavaScript的第一个异步标准模型,一个包含传递信息与状态的对象,emmm...它的英语意思是承诺.
so 它拥有的特点也有承诺的意思(兄弟们,拿好笔记本,划重点):
- 对象的状态不受外界影响,Promise 表示一个异步的操作,它有三个状态 Pending (进行时)、Reslove (已完成)、Rejected (失败),只有异步操作的结果,可以决定当前是哪一种状态
- 一旦状态改变就不会再接受改变,任何时候都可以得到这个结果。 Promise 对象的状态改变,只有两种可能,从 Pending 变为 Resolved 或者 Rejected。 只要这两种情况发生,状态就不会再改变了,就算状态改变了,你再对 Promise 对象添加回调,也会立即得到结果,这和事件 event 不同,事件的特点是一点你错过了它,再去监听也不会得到结果的
那 Promise 为什么会出现呢?
Promise 之前的异步
Promise之前的时候,我们用三个手段解决异步问题
- 回调函数
- 观察者模式(又称发布订阅模式)
- 事件机制
回调函数是最简单的异步方式,它比较容易理解和部署,也是我们最常使用的,但是缺点也很明显,不利于代码的阅读和维护,流程混乱。
观察者模式和事件机制不多说,就是适用于多次同类型的异步,不适用一次性的异步
而 Promise 可以将异步操作以同步操作的流程表达出来,
- 避免了层层回调函数。
- 如果使用回调函数, 你并不能知道对方是否执行你的回调函数,或太晚太早执行
- 错误也是异步,会隐式传递给reject
老哥,说了这么多,怎么用啊?
先来个最简单的Prmise
const p = new Promise((resolve,reject)=>{
setTimeout(()=>{
resolve('你好,兄弟') //设置成resolve状态 即是成功
},1e3)
})
p.then(res=>{
console.log(res) // 1s后输出 ==> 你好,兄弟
})
是不是看起来 so easy ?
下面我们来实现一下最简单的 Promise。 Promise只是一个普通的构造函数,用es3的语法也可以实现
所以在ie6都可以用,不过我们就使用class来实现了
class myPromise {
constructor(fn) {
this.state = 'PENDING';
this.message = '';
this.fns = [];
fn(this.resolve.bind(this), this.reject.bind(this));
}
resolve(res) {
this.state = 'FULFILLED';
this.message = res;
this._onChange();
}
reject(err) {
this.state = 'REJECTED';
this.message = err;
this._onChange();
}
then(onResolved, onRejected) {
let self = this;
return new myPromise((resolve, reject) => {
self.fns.push([onResolved, onRejected, resolve, reject]);
self._onChange();
});
}
_onChange() {
this.state !== 'PENDING' &&
this.fns.forEach(v => {
this._change(v);
});
}
_change(arr) {
let onResolved = arr[0],
onRejected = arr[1],
resolve = arr[2],
reject = arr[3];
switch (this.state) {
case 'FULFILLED':
if (typeof onResolved === 'function') {
resolve(onResolved.call(null, this.message));
} else {
resolve(this.message);
}
break;
case 'REJECTED':
if (typeof onRejected === 'function') {
resolve(onRejected.call(null, this.message));
} else {
reject(this.message);
}
break;
default:
break;
}
}
}
//老哥们注意,这只是我自己写的一个简略的 Promise , 并不是真正的 Promise 的实现
const p = new myPromise((resolve, reject) => {
setTimeout(() => {
resolve('你好,兄弟'); //1s后设置成resolve状态 即是成功
}, 1e3);
});
p.then(res => {
console.log(res); // 1s后输出 ==> 你好,兄弟
return '1';
})
.then(res => {
console.log(res); // 1s后输出 ==> '1'
})
.then(res => {
console.log(res); // 1s后输出 ==> undefined
});
Promise.resolve / reject
Promise 还提供一个快速实例化 Promise ,可以传入参数并且马上触发 Promise 链的执行
Promise.resolve = data =>
new Promise((resolve, reject) => {
resolve(data);
});
Promise.reject = err =>
new Promise((resolve, reject) => {
reject(err);
});
Promise.all / race
Promise 还提供 all / race 用来处理数据的并发和竞争,要求是传一个 Promise 数组作为参数,然后返回一个新的 Promise
all 是全部处理后再统一处理
Promise.all = arr =>
new Promise((resolve, reject) => {
let c = 0,
l = arr.length,
result = [];
l === 0 && resolve(result);
arr.forEach(pro => {
Promise.resolve(pro).then(res => {
result.push(res);
c++;
c === l && resolve(result);
},
err => {
reject(err);
}
);
});
});
race 则是谁异步时间短谁就会被处理
Promise.race = arr =>
new Promise((resolve, reject) => {
arr.forEach(pro => {
Promise.resolve(pro).then(resolve, reject);
});
});
Promise.then / catch
Promise 的 then 函数我们也知道是怎么用的,它始终以当前的结果返回一个新的Promise
而之前我们说到 Promise 的错误是异步的 也直接调用到 reject,
那我们在 resolve 或者 reject 出现错误应该怎么做呢? Promise 还提供一个catch 用来捕捉错误
Promise.catch = reject => this.then(undefined,reject)
怎么用呢?
const p = new Promise((resolve, reject) => {
//执行操作
//..略
});
p.then(resolve, reject)
.then(resolve, reject)
.then(resolve, reject)
.catch(err => {
//如果以上出现错误会直接传递到此处执行错误处理
});
Promise 和 setTimeout
说是异步,那么和我们最原始的 setTimeout 有什么区别呢?
先让我们看一道面试题目
console.log('a');
setTimeout(() => {
console.log('b');
});
console.log('c');
new Promise((resolve, reject) => {
console.log('d');
resolve('e');
}).then(res => {
console.log(res);
});
console.log('f');
输出顺序是什么呢 ?
a => c => d => f => e => b
这里就要关系到js的 EventLoop 机制了,这里并不细说,只是抛个引子,有兴趣的朋友可以去看看相关文章
结尾
到这里就结束了,老哥们,要去搬砖了
如果发现文章有什么问题,可以留言哦
随便说说Promise的更多相关文章
- 大白话讲解Promise(三)搞懂jquery中的Promise
前两篇我们讲了ES6中的Promise以及Promise/A+规范,在Promise的知识体系中,jquery当然是必不可少的一环,所以本篇就来讲讲jquery中的Promise,也就是我们所知道的D ...
- 大白话讲解Promise(一)
去年6月份, ES2015正式发布(也就是ES6,ES6是它的乳名),其中Promise被列为正式规范.作为ES6中最重要的特性之一,我们有必要掌握并理解透彻.本文将由浅到深,讲解Promise的基本 ...
- Promise学习
转自:http://www.cnblogs.com/lvdabao/p/es6-promise-1.html 去年6月份, ES2015正式发布(也就是ES6,ES6是它的乳名),其中Promise被 ...
- 你不知道的JavaScript--大白话讲解Promise
转载:http://blog.csdn.net/i10630226/article/details/50867792 一.Promise小试 复杂的概念先不讲,我们先简单粗暴地把Promise用一下, ...
- 你是否也在学习ES6 Promise时遇到过这个问题?
背景 周末闲来无事,随便翻看了一下阮一峰老师的<ES6 标准入门>第2版,ps:之前在阮一峰老师的官网看过电子版,感觉干货满满,所以就买了纸质版:当看到第16章第4节 'Promise.p ...
- Promise小书(长文)
promise基础 Promise是异步编程的一种解决方案.ES6 Promise的规范来源于Promises/A+社区,它有很多版本的实现. Promise比传统的解决方案(回调函数和事件)更合理和 ...
- Es6 Promise 用法详解
Promise是什么?? 打印出来看看 console.dir(Promise) 这么一看就明白了,Promise是一个构造函数,自己身上有all.reject.resolve这几个眼熟的方 ...
- angularjs promise详解
一.什么是Promise Promise是对象,代表了一个函数最终可能的返回值或抛出的异常,就是用来异步处理值的. Promise是一个构造函数,自己身上有all.reject.resolve这几个异 ...
- 聊一聊promise的前世今生
promise的概念已经出现很久了,浏览器.nodejs都已经全部实现promise了.现在来聊,是不是有点过时了? 确实,如果不扯淡,这篇随笔根本不会有太多内容.所以,我就尽可能的,多扯一扯,聊一聊 ...
随机推荐
- 在List中常用的linq表达式
为了下面举例方便,先声明一个集合: public List<Model.Resume> GetResumeList() { var list = new List<Model.Res ...
- 设计模式之职责链模式(JAVA实现)
学习netty框架时,看到有人说netty用到了设计模式的职责链模式,学习一下职责链模式,主要参考大话设计模式. 主要场景: 小菜想要加薪,向经理提出加薪请求,经理没有权限,经理交由总监处理,总监也没 ...
- sort遇到的问题
var arr = [2,10,6,9,7,8]; var arr1 = arr.sort(); var arr2 = arr.sort(function(a,b){ if (a>b){ ret ...
- 采用 ITextPDF 类库测试向 PDF 中加入图片的示例
package com.smbea.image; import com.artup.util.image.ImageUtil; import com.itextpdf.text.*; import c ...
- js获取css的各种样式并且设置他们
js原生获取css样式,并且设置,看似简单,其实并不简单,我们平时用的ele.style.样式,只能获取内嵌的样式,但是我们写的样式基本都在style属性里面; 这里我们就需要: 下面这个代码主要是设 ...
- python 读写Oracle10g数据简介
1.测试环境: Centos6 X86_64python 2.6 Oracle 10g 2.安装cx_Oracle 和 Oracle InstantClient: http://www.rpmfind ...
- 【Linux】动态链接函数库
动静区别 1. gcc –c mylib.c –o mylib.o 2. gcc -shared -fPIC mylib.o -o libmylib.so 3. 将制作好的libmylib.so 复制 ...
- HttpClient拉取连载小说
上午刚入手的小说,下午心血来潮想从网站上拉取下来做成电子书,呵呵,瞎折腾-说做就做- [抓包] 这一步比什么都重要,如果找不到获取真正资源的那个请求,就什么都不用做了- 先是打算用迅雷把所有页面都下载 ...
- Do the Untwist
Do the Untwist Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) ...
- JavaMelody监控spring、struts
前言 前面讲过了Javamelody的基本配置,如何使用Javamelody来监控JDBC以及SQL. 这里继续讲解一下如何监控struts和spring. 手码不易,转载请注明:xingoo 由于s ...