为啥要说 promise ?

因为这是前端必须要掌握的一个知识,吹逼必备

首先说说 Promise 是什么?

Promise 是JavaScript的第一个异步标准模型,一个包含传递信息与状态的对象,emmm...它的英语意思是承诺.

so 它拥有的特点也有承诺的意思(兄弟们,拿好笔记本,划重点):

  1. 对象的状态不受外界影响,Promise 表示一个异步的操作,它有三个状态 Pending (进行时)、Reslove (已完成)、Rejected (失败),只有异步操作的结果,可以决定当前是哪一种状态
  2. 一旦状态改变就不会再接受改变,任何时候都可以得到这个结果。 Promise 对象的状态改变,只有两种可能,从 Pending 变为 Resolved 或者 Rejected。 只要这两种情况发生,状态就不会再改变了,就算状态改变了,你再对 Promise 对象添加回调,也会立即得到结果,这和事件 event 不同,事件的特点是一点你错过了它,再去监听也不会得到结果的

那 Promise 为什么会出现呢?

Promise 之前的异步

Promise之前的时候,我们用三个手段解决异步问题

  1. 回调函数
  2. 观察者模式(又称发布订阅模式)
  3. 事件机制

回调函数是最简单的异步方式,它比较容易理解和部署,也是我们最常使用的,但是缺点也很明显,不利于代码的阅读和维护,流程混乱。

观察者模式和事件机制不多说,就是适用于多次同类型的异步,不适用一次性的异步

而 Promise 可以将异步操作以同步操作的流程表达出来,

  1. 避免了层层回调函数。
  2. 如果使用回调函数, 你并不能知道对方是否执行你的回调函数,或太晚太早执行
  3. 错误也是异步,会隐式传递给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的更多相关文章

  1. 大白话讲解Promise(三)搞懂jquery中的Promise

    前两篇我们讲了ES6中的Promise以及Promise/A+规范,在Promise的知识体系中,jquery当然是必不可少的一环,所以本篇就来讲讲jquery中的Promise,也就是我们所知道的D ...

  2. 大白话讲解Promise(一)

    去年6月份, ES2015正式发布(也就是ES6,ES6是它的乳名),其中Promise被列为正式规范.作为ES6中最重要的特性之一,我们有必要掌握并理解透彻.本文将由浅到深,讲解Promise的基本 ...

  3. Promise学习

    转自:http://www.cnblogs.com/lvdabao/p/es6-promise-1.html 去年6月份, ES2015正式发布(也就是ES6,ES6是它的乳名),其中Promise被 ...

  4. 你不知道的JavaScript--大白话讲解Promise

    转载:http://blog.csdn.net/i10630226/article/details/50867792 一.Promise小试 复杂的概念先不讲,我们先简单粗暴地把Promise用一下, ...

  5. 你是否也在学习ES6 Promise时遇到过这个问题?

    背景 周末闲来无事,随便翻看了一下阮一峰老师的<ES6 标准入门>第2版,ps:之前在阮一峰老师的官网看过电子版,感觉干货满满,所以就买了纸质版:当看到第16章第4节 'Promise.p ...

  6. Promise小书(长文)

    promise基础 Promise是异步编程的一种解决方案.ES6 Promise的规范来源于Promises/A+社区,它有很多版本的实现. Promise比传统的解决方案(回调函数和事件)更合理和 ...

  7. Es6 Promise 用法详解

     Promise是什么??    打印出来看看  console.dir(Promise) 这么一看就明白了,Promise是一个构造函数,自己身上有all.reject.resolve这几个眼熟的方 ...

  8. angularjs promise详解

    一.什么是Promise Promise是对象,代表了一个函数最终可能的返回值或抛出的异常,就是用来异步处理值的. Promise是一个构造函数,自己身上有all.reject.resolve这几个异 ...

  9. 聊一聊promise的前世今生

    promise的概念已经出现很久了,浏览器.nodejs都已经全部实现promise了.现在来聊,是不是有点过时了? 确实,如果不扯淡,这篇随笔根本不会有太多内容.所以,我就尽可能的,多扯一扯,聊一聊 ...

随机推荐

  1. 在List中常用的linq表达式

    为了下面举例方便,先声明一个集合: public List<Model.Resume> GetResumeList() { var list = new List<Model.Res ...

  2. 设计模式之职责链模式(JAVA实现)

    学习netty框架时,看到有人说netty用到了设计模式的职责链模式,学习一下职责链模式,主要参考大话设计模式. 主要场景: 小菜想要加薪,向经理提出加薪请求,经理没有权限,经理交由总监处理,总监也没 ...

  3. sort遇到的问题

    var arr = [2,10,6,9,7,8]; var arr1 = arr.sort(); var arr2 = arr.sort(function(a,b){ if (a>b){ ret ...

  4. 采用 ITextPDF 类库测试向 PDF 中加入图片的示例

    package com.smbea.image; import com.artup.util.image.ImageUtil; import com.itextpdf.text.*; import c ...

  5. js获取css的各种样式并且设置他们

    js原生获取css样式,并且设置,看似简单,其实并不简单,我们平时用的ele.style.样式,只能获取内嵌的样式,但是我们写的样式基本都在style属性里面; 这里我们就需要: 下面这个代码主要是设 ...

  6. python 读写Oracle10g数据简介

    1.测试环境: Centos6 X86_64python 2.6 Oracle 10g 2.安装cx_Oracle 和 Oracle InstantClient: http://www.rpmfind ...

  7. 【Linux】动态链接函数库

    动静区别 1. gcc –c mylib.c –o mylib.o 2. gcc -shared -fPIC mylib.o -o libmylib.so 3. 将制作好的libmylib.so 复制 ...

  8. HttpClient拉取连载小说

    上午刚入手的小说,下午心血来潮想从网站上拉取下来做成电子书,呵呵,瞎折腾-说做就做- [抓包] 这一步比什么都重要,如果找不到获取真正资源的那个请求,就什么都不用做了- 先是打算用迅雷把所有页面都下载 ...

  9. Do the Untwist

      Do the Untwist Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) ...

  10. JavaMelody监控spring、struts

    前言 前面讲过了Javamelody的基本配置,如何使用Javamelody来监控JDBC以及SQL. 这里继续讲解一下如何监控struts和spring. 手码不易,转载请注明:xingoo 由于s ...