首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

是否需要手动释放一次性定时器?

我们可以通过调用CreateTimerQueueTimer这个API来创建一个一次性定时器,只需传递一个WT_EXECUTEONLYONCE标志就行。如果你仔细阅读开发文档,则会发现文档有这么一句话:当你不再需要使用定时器时,你需要调用DeleteTimerQueueTimer来释放定时器。

问题来了:既然是一个一次性的定时器,为啥还需要手动释放它呢?

为了回答这个问题,我想向你介绍一套关于解答此类问题的一种标准技法:让我们假设,这个定时器使用之后,我们不释放它,看看会发生什么?

一方面,这会导致API的行为看起来很怪异。调用者调用CreateTimerQueueTimer函数后,需要跟踪定时器的属性(例如它是一次性的还是其他类型的),还需要知道定时器的句柄是否需要被删除。

更糟糕的是,如果一次性定时器是可以实现自动释放的,则调用者几乎不太可能正确地使用它。

假设,你有一个对象,它创建了一个一次性定时器,你希望在对象的析构函数中释放它。如果这个定时器是自动释放的,则就不可能像下图这样编写代码:

你可能会说,”我可以设计一个回调函数来将这个m_hTimer变量设置为空。这样,析构函数就知道定时器被激活过了。”

但是,这会导致一个潜在的竞争条件(Race Condition),如下图所示:

如果在竞争条件期间回调被抢占并且对象被析构,并且一次性定时器正在自动释放,那么对象将会使用无效的句柄。

这个竞争条件窗口是无法关闭的,因为竞争条件甚至在你有机会执行一行代码之前就已经发生了。

所以,请开心笑纳:你必须手动释放一次性计时器的句柄。

总结

对于实现了自动释放的对象来说,要么它完美地实现了自动化释放,这样调用者就不用太关心调用细节(干就完事了)。

要么,还是老老实实将控制权交给调用者吧。

最后

Raymond Chen的《The Old New Thing》是我非常喜欢的博客之一,里面有很多关于Windows的小知识,对于广大Windows平台开发者来说,确实十分有帮助。

本文来自:《Do you need clean up one-shot timers?》

  • 发表于:
  • 原文链接https://kuaibao.qq.com/s/20220127A07FDB00?refer=cp_1026
  • 腾讯「腾讯云开发者社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券