手写promise

写在前面:

在目前的前端分开中,我们对于异步方法的使用越来越频繁,那么如果处理异步方法的返回结果,如果优雅的进行异步处理对于一个合格的前端开发者而言就显得尤为重要,其中在面试中被问道最多的就是对Promise方法的掌握情况,本章将和大家一起分析和完成一个Promise方法,希望对你的学习有一定的帮助。

了解Promise

既然我们是要模仿ES6的Promise,那我们必然要知道这个方法主要都是用来干什么的,有哪些参数,有什么特性,为什么要使用Promise及如何使用等等。

为什么要使用它?

1.先统一执行AJAX逻辑,不关心如何处理结果,然后,在需要的时候处理AJAX结果

不知道大家有没有思考过下面的问题,JavaScript的运行都是单线程的,但是如果我们要处理类似于网络请求(ajax),浏览器的一些事件等就要用到异步执行,,大多都是下面这个样子:

function callback() {
    console.log(‘我是一个回调函数‘);
}
console.log(‘异步方法之前‘);
setTimeout(callback, 1000); // 1秒钟后调用callback函数
console.log(‘异步方法之后‘);

然后得到下面的结果:

异步操作会在将来的某个时间点触发一个函数调用,AJAX就是典型的异步操作。以jq代码为例:

$.ajax({
   type: "POST",
   url: "some.php",
   data: "name=John&location=Boston",
   success: function(msg){
     alert( "Data Saved: " + msg );
   }
});

在上面的代码中我们虽然能够得到ajax的操作结果,但是这种写法不利于我们复用,说白了异步的处理和返回结果在同一个块内,很不美观和优雅,下面来看看Promise是怎么处理这样的情况的:

let p = new Promise(function (resolve, reject) {
    setTimeout(() => {//使用定时器来模拟异步
        resolve(100)
    }, 1000);
});
p.then(function (data) {
    console.log(data)
})

可以看出p.then的调用可以是任何时候,只要我们需要时就可以拿到刚才返回结果。而不是像jq一样在ajax有结果会就要对结果进行立即处理。

2.支持链式调用

在过去,我们要进行多重异步请求的时候,一不小心就会形成回调地狱,类似于下面的这样:

doSomething(function(result) {
  doSomethingElse(result, function(newResult) {
    doThirdThing(newResult, function(finalResult) {
      console.log(‘Got the final result: ‘ + finalResult);//三次函数嵌套调用之后得到结果
    }, failureCallback);
  }, failureCallback);
}, failureCallback);

无疑,上面的函数在于阅读性和维护性上面都让我们有些力不从心,下面用Promise来实现一下上面的代码,就清晰的多:

doSomething().then(function(result) {
  return doSomethingElse(result);
})
.then(function(newResult) {
  return doThirdThing(newResult);
})
.then(function(finalResult) {
  console.log(‘Got the final result: ‘ + finalResult);
})
.catch(failureCallback);

又细心的小伙伴会发现我们的错误处理都会被集中到catch中执行,这也就是我想说的第三个特点

3.通过捕获所有的错误,promise解决了回调厄运金字塔的基本缺陷。

说了这么多,我想小伙伴已经多Promise有了一定的认识,那我就根据实际的使用,凭借自己的理解和PromiseA+规范的描述,来实现一个属于自己的promise

手写符合规范的promise

先来看代码:

let p = new Promise((resolve,reject)=>{
  resolve();
  //reject();
})

根据上面的代码我们可以看出,promise内部是一个立即执行的构造器函数,函数中有两个参数分别为resolve,reject,所以我们自己的代码应该这样写

function Promise() {
    function resolve() { }
    function reject() { }
    executor(resolve,reject)
 }

可以看到我们得到了两个函数resolve()和reject(),而且根据promiseA+规范文档中说明的:

此处我们可以得到Promise有三个状态 pending(等待状态),fulfilled(成功状态),rejected(失败状态);这个三个状态之间的关系我们用一张图来说明一下:

首先Promise在执行的时候状态都为pending,也就是等待状态,然后等待状态可以分别向成功状态和失败状态转换,但是一旦状态不是pending状态之后,这个promise的状态就无法更改,且失败状态和成功状态之间是不能相互转换的,进一步完善代码如下:

因为promise最强大的地方就在于then方法,所以不管是成功还是失败我们最终都要将成功和失败的值传递给then,为了方便调用,我们用两个变量来接收各自的值

上面已经提到promise最重要的方法就是then方法,那么为了能够在实例之后调用这个方法,我们必须将这个方法写在他的原型链上面,并且他接受两个参数,一个是成功的回调,一个是失败的回调

看下面的代码我们继续分析接下来promise是进行怎么操作的:

let p = new Promise((resolve,reject)=>{
  resolve(111);
})
p.then((value) => {
  console.log(value)
}, (reason) => {
  console.log(‘err‘, reason);
})

上面的代码最终打印结果为111,这时候我们分析在promise中如果成功了,那么then方法中的成功回调就会立即执行,如果失败了,失败的回调也会立即执行,所以我们可以继续完善我们的代码:

在上面的代码中我们只是使用同步方式,让promise函数立即执行并传入数字:111,如果是异步的情况呐?让我们进行下面的测试:

let p = new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve(111);
    }, 1000);

})
p.then((value) => {
  console.log(value)
}, (reason) => {
  console.log(‘err‘, reason);
})

这时候你会发现结果是在一秒钟之后打印出来的,也就是说,then方法中成功和失败的回调,是在promise的异步执行完成之后才被触发的,所以你在调用then方法的时候promise的状态一开始并不是成功或者失败,而是先将成功和失败的回调函数保存起来,等待异步完成之后在执行相对应的成功或者失败的回调,所以接下来我们代码可以这样写:

然后我们继续进行尝试,这次我们尝试让promise抛出一个错误看它会怎么处理?

那么反应在我们的代码中就可以这样写:

在promise中我们可以进行链式调用的方式来多次的进行then,如同下面的代码:

let p = new Promise((resolve, reject) => {
    resolve(111)

})
p.then((value) => {
  return value+‘第二次‘
}, (reason) => {
  console.log(‘err‘, reason);
    }).then((data) => {
console.log(data)
    }, () => {

    })

执行代码之后我们不难得到打印的结果为:111第二次,那么也就是如果你的then方法的成功回调函数如果返回一个值,那么我们在下一个then方法中对应的成功回调中也可以继续使用这个值,换句话说,这个值会被当作下一次then中成功回调的参数传递回来。

相同的我们测试如果出现错误的事情,会发现错误会传递给第二次的失败中

let p = new Promise((resolve, reject) => {
    resolve(111)

})
p.then((value) => {
  throw new Error()
}, (reason) => {
  console.log(‘err‘, reason);
    }).then((data) => {
console.log(data)
    }, () => {
        console.log(‘第二得到失败‘)
    })

打印结果为:第二得到失败

当然如果本次回调函数中内容为空,那么下次then中会直接走成功,而且如果是失败之后也还是可以成功的,得到结果understand,如果你不想在then方法中处理错误,你还可以使用catch方法来最终捕获错误,既然成功或者失败中可以不写参数,也就是这可以为一个空函数,也就是说then方法中的两个参数都是可选参数:

上面我们已经基本上尝试了各种返回值,那么还有一种情况也是我们需要考虑的,那就是如果返回一个promise方法会放生什么情况?

p.then(() => {
    return new Promise((resolve, reject) => {
        resolve(111)
    })
}, (reason) => {

}).then((data) => {
    console.log(‘成功了‘,data)
}, (reason) => {

})

打印结果为:成功了 111

经过尝试如果返回的是一个promise函数,那么他会等待这个promise执行完成之后在返回给下一次的then,promise如果成功,就会走下一次then的成功,如果失败就会走下一次then的失败。当然这里需要注意的是,then方法中返回的回调函数不能是自己本身,如果真的这样写,那么函数执行到里面时会等待promise的结果,这样一层层的状态等待就会形成回调地狱

现在我们的代码已经看上去原生的promise很相似了,但是为了严谨,我们进行下面的尝试:

let promise = new Promise((resolve,reject)=>{
   resolve();
});
promise.then((value) => { // pending
    return new Promise((resolve,reject)=>{
        return new Promise((resolve,reject)=>{
            resolve(111);
         })
     })
}, (reason) => {
  console.log(reason);
});

理论上我们可以得出下一次then的结果为:111,因为我们是等待promise执行完才会返回,也就是说刚才我们的代码只是判断了第一次是promise的情况,如果像上面代码的情况一样,就会出现问题,为了规避这样的问题,我们使用递归来执行:

细心的你可能发现,上面的截图中我还加入了一个called作为拦截器,那是因为如果有想我一样的小白用户,自己手写的promise是既可以成功也可以失败的,那么这里我们就要判断一下,不能让两次调用都执行,只调用第一个被调用的

这样我们的代码基本上就完美了,那我们就试一下吧:

let promise = new Promise((resolve,reject)=>{
   resolve(1);
});
promise.then((value) => { // pending
   console.log(value)
}, (reason) => {
  console.log(reason);
    });
console.log(2);

你会发现我们的执行结果是1,2,但是在本文的最开始就已经提到promise是一个处理异步的函数,执行结果应该为2,1才对,那是因为我们现在的promise的执行环境还是当前的上下文,也就是同步。做一下小小的改动,他就是异步了:

因为刚才分析得到then方法中两个回调函数可以是可选参数,所以我们也要处理一下:

扩展方法实现

因为在我们的分析中还有一个catch方法,那我们也来实现一下吧。既然是可以链式调用的方法,那我们也必须写在原型链上面:

Promise.prototype.catch = function (onrejected) {
  return this.then(null, onrejected)
}

当然promise还可以直接使用resolve()和reject()直接调用,是一种简便写法:

Promise.reject = function (reason) {
  return new Promise((resolve, reject) => {
    reject(reason)
  })
}
Promise.resolve = function (value) {
  return new Promise((resolve, reject) => {
    resolve(value);
  })
}

写在最后

至此,我们所有的promise特性就已经一一实现了,你是否已经看明白了,当然作为一个小白选手,我还有很多的不足,欢迎大家的指正,你也可以去参考promiseA+规范中的文档去看看我写的还有什么需要补充的,欢迎交流。

PS:为什么要结合promiseA+的规范?因为我们不能写一个玩具代码来应付面试考官和自己,你需要让自己的代码更具体有可读性和实用性,需要去规避可能遇到的各种因为调用而产生的问题,让你自己的代码更加无懈可击,在使用场景上也会更加丰富

原文地址:https://www.cnblogs.com/hanqingtao/p/9791255.html

时间: 2024-10-11 18:20:48

手写promise的相关文章

js手写'Promise'

/* * pending:初始化成功 * fulfilled:成功 * rejected:失败 * */ function Promise(executor) {// 执行器 this.status = 'pending'; this.value = undefined; this.reason = undefined; this.fulfilledCallback = []; this.rejectCallback = []; let resolve = (value)=>{ if(this.

手写Promise A+ 规范

基于ES6语法手写promise A+ 规范,源码实现 class Promise { constructor(excutorCallBack) { this.status = 'pending'; this.value = undefined; this.fulfilledAry = []; this.rejectedAry = []; //=>执行EXCUTOR(异常捕获) let resolveFn = result => { let timer = setTimeout(() =>

手写 Promise

在上一章节中我们了解了 Promise 的一些易错点,在这一章节中,我们会通过手写一个符合 Promise/A+ 规范的 Promise 来深入理解它,并且手写 Promise 也是一道大厂常考题,在进入正题之前,推荐各位阅读一下 Promise/A+ 规范,这样才能更好地理解这个章节的代码. 实现一个简易版 Promise 在完成符合 Promise/A+ 规范的代码之前,我们可以先来实现一个简易版 Promise,因为在面试中,如果你能实现出一个简易版的 Promise 基本可以过关了. 那

手写promise进阶版本

promise/A+规范: 术语: 1.promise是一个对象或者函数,该对象或者函数有一个then方法 2.thenable是一个函数或者对象,用来定义then方法 3.value是promise成功时的状态值 4.reason是promise失败时的状态值 要求:一.二 一. 1.三种状态 :pending | fulfilled(resolved) | rejected 2.rejected当处于pending状态的时候,可以转移到fulfilled(resolved)或者rejecte

【前端面试】同学,你会手写代码吗?

CSS 部分 两栏布局 要求:垂直两栏,左边固定右边自适应. 查看代码 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-e

22 道高频 JavaScript 手写面试题及答案

实现防抖函数(debounce) 防抖函数原理:在事件被触发n秒后再执行回调,如果在这n秒内又被触发,则重新计时. 那么与节流函数的区别直接看这个动画实现即可. 手写简化版: // 防抖函数 const debounce = (fn, delay) => { let timer = null; return (...args) => { clearTimeout(timer); timer = setTimeout(() => { fn.apply(this, args); }, del

wex5 实战 手写签名与上传

之前做过一个物流演示模块,有一个功能没做完,就是收件人收货后,可以手写签名并上传,替代传统纸张的签名.今天终于做完了. 一 效果演示: 后台图片上传成功 二 设计思路: 运用canvas插件jq-signature,制作手写签名,并通过二进制流方式上传图片 三 代码实现: 1 .下载引入插件jq-signature 手写签名插件,网上有很多,经过多种插件的逐步尝试,只有插件jq-signature达到了我想要的效果; 原因有二:1,支持手机触摸,web,鼠标.其它有的不支持手机触摸. 2,直接转

iOS开发UI基础—手写控件,frame,center和bounds属性

一.手写控件 1.手写控件的步骤 (1)使用相应的控件类创建控件对象 (2)设置该控件的各种属性 (3)添加控件到视图中 (4)如果是button等控件,还需考虑控件的单击事件等 (5)注意:View Contollor和view的关系 2.注意点 在OC开发中,Storyboard中的所有操作都可以通过代码实现,程序员一定要熟练掌握代码布局界面的能力! 设置控件监听方法的示例代码如下: [btn addTarget:self action:@selector(click:) forContro

logistic回归与手写识别例子的实现

本文主要介绍logistic回归相关知识点和一个手写识别的例子实现 一.logistic回归介绍: logistic回归算法很简单,这里简单介绍一下: 1.和线性回归做一个简单的对比 下图就是一个简单的线性回归实例,简单一点就是一个线性方程表示 (就是用来描述自变量和因变量已经偏差的方程) 2.logistic回归 可以看到下图,很难找到一条线性方程能将他们很好的分开.这里也需要用到logistic回归来处理了. logistic回归本质上是线性回归,只是在特征到结果的映射中加入了一层函数映射,