ES6-promise的简单理解

1:promise是什么

Promise 是异步编程的一种解决方案,也是为了解决回调函数层层嵌套的地狱而生

2:promise的三种状态

**1. pending (等待)

  1. fulfilled(成功的)
  2. reject(失败的)**

3:promise的特点

  1. 对象的状态不受外界影响。
  2. 一旦从等待状态变成为其他状态就永远不能更改状态了。
  3. 一旦新建Promise就会立即执行,无法中途取消
  4. 不设置回调函数callback,Promise内部抛出的错误,就不会反应到外部
  5. 当处于pending状态时,无法得知目前进展到哪一个阶段

4:promise的链式调用

  • 传统的回调地狱,代码少还可以,多了简直无法看了
// 模拟发送请求

let req = function(callback) {
  setTimeout(() =>{
    let data = 123;
    callback(data)
  },500)
}
// 嵌套函数的回调地狱
req((data) =>{
  console.log("1",data); // 1 123
  req((data)=>{
    console.log("2",data); // 2 123
    req((data)=>{
      console.log("3",data); // 3 123
      req((data)=>{
        console.log("4",data); // 4 123
        req((data)=>{
          console.log("5",data); // 5 123
        })
      })
    })
  })
})
 
  • promise的写法,相对于回调来说就简介优雅一些
let p1 = new Promise((resolve,reject)=>{
  setTimeout(()=>{
    resolve(1)
  },100)
})
p1.then(res =>{
  console.log("res",res); // 1
  return res
}).then(res=>{
  console.log("1",res); // 1
  return res 
}).then(res=>{
  console.log(res); // 1
})

5:promise之return

  • return 一个promise对象

let i = 0;
let fn = function() {
  return new Promise((resolve,reject)=>{
    setTimeout(()=>{
      resolve(i++)
    },100)
  })
}
fn().then(res =>{
  console.log("1",res);
  return fn() // 每次是会返回一个新的promise 0 
}).then(res =>{
  console.log("2",res);
  return fn() // 每次都返回新的promsie 1
}).then(res =>{
  console.log("3",res); 2
})

可以看出每次执行一个新的promise都会返回一个新值


在这里插入图片描述
  • promise之return 一个值
let i = 0;
let fn = function() {
  return new Promise((resolve,reject)=>{
    setTimeout(()=>{
      resolve(i++)
    },100)
  })
}
fn().then(res =>{
  console.log("1",res);
  return res
}).then(res =>{
  console.log("2",res);
  return res
}).then(res =>{
  console.log("3",res); // 返回最初的值 0
})

可以看出每次return传递值都是上次返回的结果值


在这里插入图片描述

6:promise.all(处理并发请求)

  • promise.all的普通写法
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},300)
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ res("000")},300)
})

let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
})

let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(333)},300)
})

let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
})



let arr = [p1,p2,p3,p4,p5]

Promise.all(arr).then(res=>{
  console.log('all====res'+res)
}).catch(err=>{
  console.log('err===='+err)
})

可以看出五个结果值都执行了

在这里插入图片描述

  • promise.all异常会怎样.,我们把p2对象rej一个错误值
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},300)
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej("000")},300)
})

let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
})

let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(333)},300)
})

let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
})
let arr = [p1,p2,p3,p4,p5]

Promise.all(arr).then(res=>{
  console.log('all====res'+res)
}).catch(err=>{
  console.log('err===='+err)
})

可以看出只执行了错误的值,这在实际开发中,显然不是我们想要的结果值,我们想要的是失败的或者成功的都显示出来

在这里插入图片描述

  • promise.all,处理异常,我们把p2和p4对象分别返回一个错误值
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},300)
}).then(res=>{
  console.log("res-111",res);
}).catch(err=>{
  console.log("err-111",err);
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej("000")},300)
}).then(res=>{
  console.log("res-000",res);
}).catch(err=>{
  console.log("err-000",err);
})


let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
}).then(res=>{
  console.log("res-222",res);
}).catch(err=>{
  console.log("err-222",err);
})


let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(333)},300)
}).then(res=>{
  console.log("res-333",res);
}).catch(err=>{
  console.log("err-333",err);
})


let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
}).then(res=>{
  console.log("res-444",res);
}).catch(err=>{
  console.log("err-444",err);
})


let arr = [p1,p2,p3,p4,p5]

Promise.all(arr).then(res=>{
  console.log('all====res'+res)
}).catch(err=>{
  console.log('all====err'+err)
})

显然这种结果是我们想要的,但是如果有几十条几百条,这样写是不是太费手了

在这里插入图片描述

  • promise.all,批量处理异常
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},300)
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej("000")},300)
})

let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
})

let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej(333)},300)
})

let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
})



// watchPromiseError 统一处理成功或者错误的
async function watchPromiseError(p) {
  try {
    console.log("p",p);
    const data = await p;
    return {
      err:0,
      res: data
    }
  } catch (error) {
    return {
      err: 1,
      res:error
    }
  }
}

let arr = [p1,p2,p3,p4,p5]
Promise.all(arr.map(item => watchPromiseError(item))).then(res=>{
  console.log("promiseAll-res",res);
}).catch(err=>{
  console.log("promiseAll-err",err);
})

可以看出这样也能拿到所有的结果值,还有一种promsie.allSettled也可以解决

在这里插入图片描述

6:promsie.allSettled(批量处理请求)

  • allSettled和all类似,不同的是,allSettled可以返回成功和失败的结果,不会再失败后被中断
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},300)
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej("000")},300)
})

let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
})

let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej(333)},300)
})

let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
})



// watchPromiseError 统一处理成功或者错误的
async function watchPromiseError(p) {
  try {
    console.log("p",p);
    const data = await p;
    return {
      err:0,
      res: data
    }
  } catch (error) {
    return {
      err: 1,
      res:error
    }
  }
}

let arr = [p1,p2,p3,p4,p5]
Promise.allSettled(arr).then(res=>{
  console.log("allSettled-res",res);
}).catch(err=>{
  console.log("allSettled-err",err);
})
在这里插入图片描述

promise.race

  • 普通写法
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},500)
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ res("000")},300)
})

let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
})

let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(333)},300)
})

let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
})
let arr = [p1,p2,p3,p4,p5];

Promise.race([p1,p2,p3,p4,p5]).then(res=>{
  console.log(res); // 111
})

可以看出谁先执行就先返回谁

在这里插入图片描述

  • race异常是怎样的呢,我们把p2改为rej看下
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},500)
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej("000")},300)
})

let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
})

let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(333)},300)
})

let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
})
let arr = [p1,p2,p3,p4,p5];

Promise.race([p1,p2,p3,p4,p5]).then(res=>{
  console.log(res); // 111
})

可以看出报错了,被reject拒绝了,值是000,吧啦吧啦的,反正就这个报错意思吧

在这里插入图片描述

  • 处理异常
let p1 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(111)},500)
})

let p2 = new Promise((res,rej) =>{
  setTimeout(()=>{ rej("000")},300)
})

let p3 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(222)},300)
})

let p4 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(333)},300)
})

let p5 = new Promise((res,rej) =>{
  setTimeout(()=>{ res(444)},300)
})
let arr = [p1,p2,p3,p4,p5];
async function watchPromiseError(p) {
  try {
    const data = await p;
    return {
      err:0,
      res: data
    }
  } catch (error) {
    return {
      err: 1,
      res:error
    }
  }
}
Promise.race(arr.map(item=>watchPromiseError(item))).then(res=>{
  console.log(res); // 111
})

看出就算出错,也是只返回最先请求的那个方法,正如race本身字面来说,就是赛跑的意思,谁跑的快就是谁

在这里插入图片描述

7:promise.resolve和reject

let p2 = new Promise((resolve,reject) =>{
  setTimeout(()=>{ reject("000")},300)
})


p2.then(res=>{
  console.log(res);
}).catch(err=>{
  return Promise.resolve(err) // 相当于 resolve了
}).then(res=>{
  console.log("res",res); // 000
  return Promise.reject(res) // 相当于reject了
}).then(res=>{
  console.log(res);
}).catch(err=>{
  console.log("err",err); // 000
})

在这里插入图片描述

8:手写简单的promise

class MyPromise {
  constructor(executor) {
    // 规定状态
    this.state = "pending"
    // 保存 `resolve(res)` 的res值
    this.value = undefined
    // 保存 `reject(err)` 的err值
    this.reason = undefined
    // 成功存放的数组
    this.successCB = []
    // 失败存放的数组
    this.failCB = []


    let resolve = (value) => {
      if (this.state === "pending") {
        this.state = "fulfilled"
        this.value = value
        this.successCB.forEach(f => f())
      }
    }
    let reject = (reason) => {
      if (this.state === "pending") {
        this.state = "rejected"
        this.value = value
        this.failCB.forEach(f => f())
      }
    }

    try {
      // 执行
      executor(resolve, reject)
    } catch (error) {
      // 若出错,直接调用reject
      reject(error)
    }
  }
  then(onFulfilled, onRejected) {
    if (this.state === "fulfilled") {
      onFulfilled(this.value)
    }
    if (this.state === "rejected") {
      onRejected(this.value)
    }
    if (this.state === "pending") {
      this.successCB.push(() => { onFulfilled(this.value) })
      this.failCB.push(() => { onRejected(this.reason) })
    }
  }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容