JS实现"上次操作未完成禁止新操作"逻辑特事特办方案

 更新时间:2023年05月16日 11:27:31   作者:知名喷子  
这篇文章主要介绍了详解JS如何实现"上次操作未完成禁止新操作"的逻辑及思路,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

场景

相信很多人都遇到过类似的场景:

某一个按钮是用来发送请求的,并且需要一段时间来处理。但是用户往往会在处理期间有意或无意地点击多次,因此我们希望在上一次发出的请求处理完毕之前,不再发出新的请求。

1.初步解决方案:特事特办

“特事特办”的意思,就是每次遇到这样的场景,都特意写一段逻辑来处理:

document.addEventListener('click', (() => {
    let lock = false;
    return () => {
        if(lock) return;
        lock = true;
        console.log('clicked');
        // 为了方便测试就使用延时了
        setTimeout(() => {
            lock = false;
        }, Math.random() * 4e3)
    }
})());

2. 基于约定回调的条件式回调函数

上面的写法其实也不费事,但是这种条件限制能不能像已经被面试问烂了的“节流”和“防抖”那样,用一个函数把它包裹起来就可以达成效果呢?
问题的关键其实在于:防抖和节流需要考虑的执行条件是时间,这个条件对于所有函数而言都是一个“共同的语言”,因此才双方可以做到那样的“默契”。
而要在这种场景里实现同样的效果,双方需要刻意的约定:例如被条件执行的函数额外接受一个函数,用于在合适的时机解除条件限制。

function conditioned(callback:(release:Function,...args:any[]) => any){
    let lock = false;
      return function(...args:any[]){
        if(lock) return;
        lock = true;
        callback.call(this, () => {
              lock = false;
        }, ...args);
      }
}

 👆 为了方便描述这种约定对callback 的要求,这里使用 TS 而不是 JS。

使用的时候:

button.addEventListener('click', conditioned((release) => {
    return () => {
        console.log('clicked');
        setTimeout(() => {
            release(); // 释放 lock
        }, Math.random() * 4e3);
    }
}));

3. 基于 Promise 的条件式回调函数

如果说有什么方法能比回调函数更“优雅”、更“通用”一些,答案显然是Promise

因为上面的写法对原来的回调函数的参数进行了改写,遇到一个爱好 Ctrl + C 的初学者的话,他会疑惑复制过来的函数为什么就不工作了。

function conditioned(callback:(...args:any[]) => Promise<any>){
    let lock = false;
    return function(...args:any[]){
        if(lock) return;
        lock = true;
        try {
            await callback.call(this, ...args);
            lock = false;
        } catch(err) {
            lock = false;
            throw err;
        }
    }
}

使用方法:

button.addEventListener('click', conditioned(() => {
  return new Promise((resolve) => {
      console.log('clicked');
      setTimeout(() => {
          resolve(); // 释放 lock
      }, Math.random() * 4e3);
  });
}));

虽然乍一看,使用这个函数意味着必须把回调函数的返回值改写成 Promise,不过由于这种场景往往都是异步操作,改成 async 何乐而不为呢?

4. React hook 版

import { useRef } from 'react';
function useCondition(callback: (...args: any[]) => Promise<any>) {
  const lock = useRef(false);
  return async (...args:any[]) => {
    if(lock.current) return;
    lock.current = true;
    try{
      await callback(...args);
      lock.current = false
    } catch(error){
      lock.current = false;
      throw error;
    }
  };
}

使用方法:

<button
  onClick={useCondition(() => {
    return new Promise<void>((resolve) => {
      console.log('clicked');
      setTimeout(() => {
          resolve(); // 释放 lock
      }, Math.random() * 4e3);
    });
  })}
>测试</button>

 👆实际上没测试过,不知道行不行的,更多关于JS 上次未完成禁止新操作的资料请关注脚本之家其它相关文章!

相关文章

  • JS动画效果打开、关闭层的实现方法

    JS动画效果打开、关闭层的实现方法

    这篇文章主要介绍了JS动画效果打开、关闭层的实现方法,可实现js控制层从中心位置打开与关闭的功能,涉及javascript操作页面元素的相关技巧,需要的朋友可以参考下
    2015-05-05
  • javascript实现多边形碰撞检测

    javascript实现多边形碰撞检测

    这篇文章主要介绍了javascript如何实现多边形碰撞检测,帮助大家更好的理解和使用js,感兴趣的朋友可以了解下
    2020-10-10
  • 教你用wxml2canvas将微信小程序页面转为图片

    教你用wxml2canvas将微信小程序页面转为图片

    如果需要实现将小程序的页面转为图片,第一步是要先把页面转为canvas,再将canvas转为图片,下面这篇文章主要给大家介绍了关于用wxml2canvas将微信小程序页面转为图片的相关资料,需要的朋友可以参考下
    2022-11-11
  • 让你的网站可编辑的实现js代码

    让你的网站可编辑的实现js代码

    可以让你编辑浏览器中看到网页可编辑的实现代码。
    2009-10-10
  • D3.js 实现带伸缩时间轴拓扑图的示例代码

    D3.js 实现带伸缩时间轴拓扑图的示例代码

    这篇文章主要介绍了D3.js 实现带伸缩时间轴拓扑图的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-01-01
  • 基于rollup的组件库打包体积优化小结

    基于rollup的组件库打包体积优化小结

    这篇文章主要介绍了基于rollup的组件库打包体积优化小结,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-06-06
  • js常用函数2008-8-16整理

    js常用函数2008-8-16整理

    方便使用js的朋友,把下面的函数找到你想要的功能函数,复制部分判断输入文本是否为身份证号码,如为不正确则提示
    2008-08-08
  • 微信小程序实现滑动侧边栏

    微信小程序实现滑动侧边栏

    这篇文章主要为大家详细介绍了微信小程序实现滑动侧边栏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-07-07
  • Echarts柱状图修改柱子颜色渐变及柱子圆角效果实例

    Echarts柱状图修改柱子颜色渐变及柱子圆角效果实例

    在ECharts中,可以通过修改series.itemStyle.normal属性来定制柱状图的柱子样式,包括圆角和背景色渐变,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-09-09
  • js中split函数的使用方法说明

    js中split函数的使用方法说明

    本篇文章主要是对js中split函数的使用方法进行了说明介绍,需要的朋友可以过来参考下,希望对大家有所帮助
    2013-12-12

最新评论