Promise解决实际问题

2022-09-07  本文已影响0人  Poppy11

1、取消重复请求

场景:一般点击了A按钮,请求已经发送,请求还没完的时候,再次点击按钮,又会发送新的请求。此场景,解决办法也有很多,例如loading效果,或者使用react-query库。在此我们使用Promise的特性解决

方案:创建一个类,使用Promise.race()的特性,每次请求时,在当前请求身旁埋一个雷,也就是说每次发送请求时,创建一个Promise数组,数组里面包含当前请求的Promise和自己新建的Promise,如果说在此发送请求,发现promise正在请求中,就把这个请求炸掉,开启新的请求

cancelAblePromise.ts
export default class CancelAblePromise {
    pendingPromise : Promise<any> | null,
    reject : Function;
     
    constructor() {
        this.pendingPromise = null;
        this.reject = () => {};
    }

    request(requestFn : Function){
        if(this.pendingPromise){
          this.cancel('取消请求');
        }
        
        const cancelPromise = new Promise((resolve,reject) => this.reject = reject);
        this.pendingPromise = Promise.race([requestFn,cancelPromise]);
        return this.pendingPromise;
    }

    cacel(reason : string){
        this.reject(reason);
        this.pendingPromise = null;
    }
}

test.ts
useEffect(() => {
    request();
  }, []);

  const requestDemo = () => {
    return new Promise((resolve, reject) => {
      setTimeout(() => {
        resolve('赢了');
      }, 2000);
    });
  };

  const request = async () => {
    const cancelPromise = new cancelAblePromise();
    for (let i = 0; i < 5; i++) {
      cancelPromise
        .request(requestDemo)
        .then((res) => console.log(res))
        .catch((error) => console.log(error));
    }
  };

2.超时控制

场景:转盘问题,一个抽奖转盘动画效果有5秒,但是一般来说向后端请求转盘结果只需要不到一秒,因此请求结果至少得等5秒才能展现给用户。
1.转盘动画还未完成,请求结果已经拿到了,此时要等到动画完成再展示结果给用户。
2.转盘动画完成了,请求结果还未拿到,此时需要等待结果返回(可以设置请求超时时间)。

思路:设置个超时的Promise,使用Promise.race,看Request和超时Promise,谁先返回结果。

function sleep(delay){
  return new Promise((resolve,reject) => {
    setTimeout(() => {reject('超时')},delay)
  })
}

function timeoutPromise(requestFn,timeout){
  return Promise.race([requestFn,sleep(timeout)]);
}

-----自定义实现Promise.race------
function timeoutPromise(requestFn,timeout){
  const promises = [requestFn(),sleep(timeout)];
  return new Promise((resolve,reject) => {
    for(const promise of promises){
      promise.then(res => resolve(res)).catch(error => reject(error));
    }
  })
}


3.Promise实现串行

思路:
1.串行就需要Promise依据上一次的结果一直then下去
2.这一次需要依赖上一次的值考虑使用reduce

function execute(tasks) {
    return tasks.reduce(
    (previousPromise, currentPromise) => previousPromise.then((resultList) => {
        return new Promise(resolve => {
            currentPromise().then(result => {
                resolve(resultList.concat(result))
            }).catch(() => {
                resolve(resultList.concat(null))
            })
        })
    }),
    Promise.resolve([])
    )
}
上一篇 下一篇

猜你喜欢

热点阅读