防止重复执行、重复点击、重复请求

文章介绍了如何使用Promise和Map来防止在指定函数产生结果前重复运行,适用于接口请求的防抖场景,同时支持异步和普通函数。示例展示了如何在点击事件中应用该功能,确保多次点击只执行一次函数并共享结果。
摘要由CSDN通过智能技术生成

应用场景:在指定函数产生结果之前,防止重复运行指定的函数,比如,接口重复请求。

/***
 * 在指定函数产生结果之前,防止重复运行指定的函数
 */
function usePrmoiseResult() {
    const map = new Map()

    /**
     * 指定函数产生结果之前,防止重复运行指定函数
     * @param fn 指定函数,可以是一个async异步函数
     * @param key 防止重复运行的唯一值
     * @returns 
     */
    const awaitPrmoiseResult = async function <T>(fn: () => T, key: any): Promise<T> {
        if (!map.has(key)) {
            map.set(key, Promise.all([fn()]))
        }
        return map.get(key).then((result: any[]) => {
            return result[0]
        }).finally(() => {
            map.delete(key)
        })
    }

    const removePrmoiseResult = function (key: any) {
        map.delete(key)
    }

    const clearPrmoiseResult = function () {
        Array.from(map.keys()).forEach((key: any) => {
            map.delete(key)
        });
    }

    return {
        awaitPrmoiseResult,
        removePrmoiseResult,
        clearPrmoiseResult
    }
}

使用案例1(异步函数):

const { awaitPrmoiseResult } = usePrmoiseResult()

window.onclick = () => {
    const fn = async () => {
        console.log("运行指定函数", new Date().toLocaleTimeString())
        await awaitTime(5000)
        return 123
    }
    awaitPrmoiseResult(fn, "asdsds").then((res: any) => {
        console.log("产生结果了", res, new Date().toLocaleTimeString())
    })
}

连续点击6次,打印结果如下:

虽然连续点击了6次,但指定函数只执行一次,并且函数执行的结果共享,作为了其他相同key的指定函数的结果。

  • 是不是 像 相同接口重复请求,多个相同请求共享一份数据的场景呢!!!他可以使用在请求当中去。
  • 是不是可以做为立即运行的防抖函数呢!!!防止重复点击。

使用案例2(普通函数):

const { awaitPrmoiseResult } = usePrmoiseResult()

window.onclick = () => {
    const fn = () => {
        console.log("运行指定函数", new Date().toLocaleTimeString())
        return 123
    }
    awaitPrmoiseResult(fn, "asdsds").then((res: any) => {
        console.log("产生结果了", res, new Date().toLocaleTimeString())
    })
}

连续点击6次,打印结果如下:

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值