promise是用来解决Js中的异步问题的,js中所有的异步可从callback → promise → generator + co = async + await

其实所有的都是callback的变相处理,只是后者慢慢变的越来越优雅和强壮可扩展。

那么如何实现promise呢?先观察一下promise的样子

let  a = new Promise((resolve,reject)=>{
// dosomething
resolve()
})

无非是一个名称叫Promise的对象,然后传参一个函数 (resolve,reject)=>{resolve()}

promise的强大之处在于回调处理,promise.then(resolveFunticon,rejectFunction)

第一个函数是resolve的回调处理函数,第二个是reject

说到resovle,reject,那么还得再提一下pending,promise对象内部有以上三种状态。默认是pending状态

状态的转变需要我们手动自己去调用函数resolve()或者reject()

整理一下思路:new promise的时候callback作为传参参数执行promise函数,callback在promise函数内部执行。callback里手动调用的resolve或者reject方法,那么promise对象里肯定是有这两种方法的。

最关键的是记住状态,然后再then的时候根据状态去判断到底是执行resolveFunticon还是rejectFunction。

function PromiseFn(callBack){
let self = this;
self.resolveVal = undefined;
self.rejectVal = undefined;
self.status = 'pending'//默认是等待状态 function resolve(val){
if(self.status === 'pending'){
self.resolveVal = val;
self.status = 'resolve'
}
}
function reject(val){
if(self.status === 'pending'){
self.rejectVal = val;
self.status = 'reject'
}
}
     //这里要注意,try catch若是写在 let self = this 会报错,let存在暂时死区,没有常规的变量提升。必须先申明后使用
     // callback执行,调用resolve函数。
try{
callBack(resolve,reject)//callback进来自动执行
}catch(e){
reject()//若是有报错,则直接变为reject状态
}
}

上面关于let的深入可以去看这篇博客:我用了两个月的时候才理解let

可能会被callback给弄得有点懵,其实把它当做函数fn就行,fn在promise内部执行,然后fn内部刚好有个resolve(),然后再去调用promise的这个resolve执行。

也就是这个原因,所以callback的参数必须是resolve和reject。

上面这一步,需要注意的是callback是立即执行的。下面这段会立即打印111

let  a = new Promise((resolve,reject)=>{
console.log('111')
})
console.log('456')

上面我们定义了 resolveVal 和 rejectVal,因为待会在promise调用then执行的时候,会作为参数传参给resolveFunticon或者rejectFunction

所以promise对象内部必须记住这个参数。

下面一起看看promise.then吧,promise.then(resolveFunticon,rejectFunction)。在promise的原型链上有个then方法

PromiseFn.prototype.then = function(resolveFunction,rejectFunction){
  let self = this;
  if(self.status === 'resolve'){
resolveFunction(self.resolveVal)
  }
  if(self.status === 'reject'){
rejectFunction(self.rejectVal)
  }
}

在then执行的时候,去判断内部是啥状态,然后执行对应的resolve或者reject回调函数。  

然后运行试一试

let promiseA = new PromiseFn((resolve,reject)=>{
resolve('成功啦')
})
promiseA.then((resolveVal)=>{
console.log('resolve'+resolveVal)
},(rejectVal)=>{
console.log('reject'+rejectVal)
})

结果是:  

但是这样就结束了吗?no,我们一般对Promise的使用场景可能是下面这样的,需要考虑异步问题。

let promiseA = new PromiseFn((resolve,reject)=>{
setTimeout(function(){
resolve('成功啦')
},2000)
})

这里用settimeout代替一下,一般我们都是一个ajax请求,然后在请求到结果后去resolve.

上面的写法显然是无法满足这种条件的,当去执行promise.then的时候,发现状态还是pending,不做任何执行。

那我们怎么去做呢?发现状态是pending,肯定是两秒钟之内啦,不妨先把promise.then两个函数参数先在promise里面记下来,两秒钟后状态变为resolve了,再执行函数。

      function PromiseFn(callBack){
try{
callBack(resolve,reject)
}catch(e){
reject()
}
let self = this;
self.resolveVal = undefined;
self.rejectVal = undefined;
self.status = 'pending'//默认是等待状态
self.keepResolveFn = []//
self.keepRejectFn = []// function resolve(val){
if(self.status === 'pending'){
self.resolveVal = val;
self.status = 'resolve';
self.keepResolveFn.forEach(fn=>fn());
}
}
function reject(val){
if(self.status === 'pending'){
self.rejectVal = val;
self.status = 'reject';
self.keepRejectFn.forEach(fn=>fn());
}
}
//执行先记录resolve和reject函数事件 }
PromiseFn.prototype.then = function(resolveFunction,rejectFunction){
let self = this;
if(self.status === 'resolve'){
resolveFunction(self.resolveVal)
}
if(self.status === 'reject'){
rejectFunction(self.rejectVal)
}
if(self.status === 'pending'){
self.keepResolveFn.push(()=>{
resolveFunction(self.resolveVal);
});
self.keepRejectFn.push(()=>{
rejectFunction(self.rejectVal)
});
}
}
let promiseA = new PromiseFn((resolve,reject)=>{
setTimeout(function(){
resolve('成功啦')
},2000)
})
promiseA.then((resolveVal)=>{
console.log('resolve'+resolveVal)
},(rejectVal)=>{
console.log('reject'+rejectVal)
})

结果是:延迟两秒后,打印出  ‘resolve成功啦’。

最后多提一句:所有的then方法执行,也是通过return 一个新的promise对象来执行的。这样才有链式回调,new promise.then().then().then()

        var b = 10;
new Promise((resolve, reject) => { setTimeout(() => {
b += 10;
resolve();
// reject();
}, 1000); }).then(function() {
console.log(b);
return new Promise((resolve, reject) => {
setTimeout(() => {
b *= 2;
resolve();
}, 1000);
});
}).then(function() {
console.log(b);
return new Promise((resolve, reject) => {
setTimeout(() => {
b *= b;
resolve();
}, 1000);
});
}).then(function() {
console.log(b);
},function() {
console.log('失败了');
});

  

  

 

JS实现Promise原理的更多相关文章

  1. Promise原理 && 简单实现

    Promise原理 参考https://github.com/chunpu/promise/blob/master/promise.js 个人认为原博的实现有点问题 在next函数的实现上, 会导致无 ...

  2. Promise 原理

    异步:可同时好几件事,互不影响: 同步:按循序一件一件.... 异步好多缺点:.... promise就是解决异步计算的这些缺点的,主要用于: 1.异步计算: 2.可以将异步操作队列化  按期望的顺序 ...

  3. Sea.Js的运行原理(转)

    1.CMD(Common Module Definition)规范 Sea.js采用了和Node相似的CMD规范,使用require.exports和module来组织模块.但Sea.js比起Node ...

  4. 深入研究Node.js的底层原理和高级使用

    深入研究Node.js的底层原理和高级使用

  5. promise原理

      简介 Promise 对象用于延迟(deferred) 计算和异步(asynchronous )计算.一个Promise对象代表着一个还未完成,但预期将来会完成的操作.Promise 对象是一个返 ...

  6. JavaScript是如何工作的: CSS 和 JS 动画底层原理及如何优化它们的性能

    摘要: 理解浏览器渲染. 原文:JavaScript是如何工作的: CSS 和 JS 动画底层原理及如何优化它们的性能 作者:前端小智 Fundebug经授权转载,版权归原作者所有. 这是专门探索 J ...

  7. vue.js响应式原理解析与实现

    vue.js响应式原理解析与实现 从很久之前就已经接触过了angularjs了,当时就已经了解到,angularjs是通过脏检查来实现数据监测以及页面更新渲染.之后,再接触了vue.js,当时也一度很 ...

  8. 深入解析vue.js响应式原理与实现

    vue.js响应式原理解析与实现.angularjs是通过脏检查来实现数据监测以及页面更新渲染.之后,再接触了vue.js,当时也一度很好奇vue.js是如何监测数据更新并且重新渲染页面.vue.js ...

  9. 浅谈Generator和Promise原理及实现

    Generator 熟悉ES6语法的同学们肯定对Generator(生成器)函数不陌生,这是一个化异步为同步的利器. 栗子: function* abc() { let count = 0; whil ...

随机推荐

  1. JAVA高并发秒杀API项目的学习笔记

    一步一步的搭建JAVA WEB项目,采用Maven构建,基于MYBatis+Spring+Spring MVC+Bootstrap技术的秒杀项目学习的视频:http://www.imooc.com/l ...

  2. Python 多继承(新式类) 的mro算法

    转载自:http://www.cnblogs.com/panyinghua/p/3283831.html mro即method resolution order,主要用于在多继承时判断调的属性的路径( ...

  3. linux 防止ssh暴力破解密码

    收集 /var/log/secure 里面的信息,若是某个IP 链接次数超过一定次数 ,则把此ip记录到/etc/hosts.deny里面 #!/bin/bash #Denyhosts SHELL S ...

  4. 用javascript实现禁止页面后退返回上一页的代码

    用javascript实现禁止页面后退返回上一页的代码:  有时候我们需要用户在点击了如下一步的按钮时,页面跳转到了下一个页面,这时想不允许用户返回后退到上一页,可以采用下面的方法:  在需要跳转的页 ...

  5. android 官网访问不了

    网上搜到的解决方案,亲测有用.记下来,以备遗忘. 使用管理员权限,修改C:\Windows\System32\Drivers\etc\hosts文件,加入以下内容 173.194.127.7 deve ...

  6. 轮播图jq版

    轮播图的需求 1:图片自己轮播,并且下面的tabs小图标跟着显示 2.鼠标hover到tabs上面显示对应的图片 3.点击左右按钮,显示下一张图片. <div id="oLunbo&q ...

  7. C++/CLI中的const literal initonly 友元(转)

    C++/CLI中的const literal initonly 友元 C++/CLI中的const Visual C++并不允许在const对象上调用成员函数,除非该成员函数也被声明为const. C ...

  8. intellijidea课程 intellijidea神器使用技巧2-1 无处不在的跳转

    idea快捷键(基于windows平台) 1 书签跳转 Ctrl alt [ ]   ==> 项目之间的跳转 Ctrl shift E ==> 文件之间的跳转(最近编辑的文件) Ctrl ...

  9. pdf word excel ppt 在线预览方案收集

    https://www.idocv.com/docs.html http://www.cnblogs.com/wolf-sun/p/3569960.html http://coolwanglu.git ...

  10. IntelliJ、ReSharper 4折 加入慧都“惊喜惠”

    慧都2013岁末回馈惊喜不断!著名的软件开发公司JetBrains旗下所有产品加入"惊喜惠"活动环节, JAVA IDE——IntelliJ IDEA,.NET效率工具集——ReS ...