JS是基于什么考虑不提供当前所有正在执行的定时任务的句柄?

定时句柄会拿不到
为什么不出个getAllInterval,getAllTimeout之类的函数?
是因为安全问题, 性能问题还是别的原因不提供的?

阅读 2.7k
4 个回答

为什么?也许他们就是单纯地没想到

JS 中欠考虑的设计多了去了,所以很多问题你与其思考设计者为什么“考虑不”,不如在心中默认他们是“不考虑”。

题目里说的那种接口可能存在什么问题

在我看来,现有的setTimeout这种以简单类型作为任务标识的接口确实是有一定的隐患的,恶意开发者可以非常轻易地干预其他模块的延时任务,如果再定制一个更好使的接口出来,那无异于是给恶意代码递棍子(为什么不是递刀子呢?因为这个隐患其实不大,基本是安全的)。
JS 虽然没有提供这样的接口,但是据我的经验,主流浏览器里面浏览器定时器/延时器返回的都是递增的整数,所以我们可以通过枚举的方法来干预页面里的所有定时/延时任务
比如下面的代码就可以枚举取消浏览器中既有的全部延时任务,在一些使用延时器做动画或其他调度的页面上运行,可能会导致页面出 bug :

const MAX_HANDLER_ID = setTimeout(_=>_);
for(let i = 0; i <= MAX_HANDLER_ID; i++){
  clearTimeout(i);
}

下面这段代码则可以把 Bilibili 首页的轮播图暂停下来:

const MAX_HANDLER_ID = setInterval(_=>_);
for(let i = 0; i <= MAX_HANDLER_ID; i++){
  clearInterval(i);
}
👆你看,因为接口是这样设计的,所以不用他们给我递棍子,我还是可以整点花活,只不过这只能算一个花活,没有任何危险。

不过制定规范的人显然对这种事情不以为意,不然后面出的 requestAnimationFrame 接口就应该返回 Symbol ,以防此类枚举。

唉,这和什么JS不JS的无关吧?你用啥语言做个定时任务还有个句柄的?充其量你写这个定时啥的就是个函数。

举个很简单的例子,PHP中你要做延时就是加一个sleep(10)(延迟10秒),golang中暂停执行也是:time.Sleep,哪里还提供啥句柄给你,你自己处理的定时就自己想办法通过其他方式解决掉取消定时这个问题,根本就没有说给你个句柄做啥取消。

JS提供给你了,让你给方便取消啥的,你还想要个获取所有定时句柄的操作,另外你自行想一下,如果要提供给你所有句柄,让你去写这么一个getAllInterval或者getAllTimeout你怎么去实现?

所以如果你想,你自己写一个呗,再封装一层函数,类似于:

var allIntervalHandle = [];
function customSetInterval(callback,seconds) {
    var handle = setInterval(callback,seconds);   
    allIntervalHandle.push(handle)
}

function getAllInterval() {
    return allIntervalHandle;
}

定时器不是会返回句柄/唯一标识吗, 你想实现这个功能可以自己做

const hwnd = setTimeout(()=>{}, 10000)
const hwnd = setInterval(()=>{}, 1000)
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
宣传栏