您当前的位置: 首页 > 

寒冰屋

暂无认证

  • 0浏览

    0关注

    2286博文

    0收益

  • 0浏览

    0点赞

    0打赏

    0留言

私信
关注
热门博文

使用CancellationToken——而不是Thread.Sleep

寒冰屋 发布时间:2020-06-04 21:28:14 ,浏览量:0

目录

介绍

使用代码

兴趣点

经常发现需要在代码执行中添加暂停,乍一看远程目标或文件可能尚未准备好,所以您退后一段时间再试一次,重复执行直到目标准备好或过程被取消。在这种情况下,很容易放入Thread.Sleep(n)——但这不会响应取消触发器,所以为什么不使用CancellationToken呢?

  • 下载源代码-10.9 KB
介绍

不久前,我被要求查看一项服务,该服务间歇性地在停止和关闭操作上生成异常。

该服务本身是一个简单的设备管理组件,它使用一个任务在启动时连接到远程设备,进行一些更新,仅此而已。但是,当尝试进行连接时,目标设备并不总是准备就绪,因此使用了while循环,并以调用Thread.Sleep(20000)的形式插入了暂停。

在再次尝试之前将任务暂停20秒的想法很合理——但是20秒线程块会间歇性地(取决于时间)导致服务管理器由于触发关闭或停止超时而强制关闭服务,由此产生的异常将填满日志。

Thread.Sleep(n)无法取消——请考虑使用CancellationToken.WaitHandle.WaitOne(n)。

使用代码

本技巧中的代码是Thread.Sleep和CancellationToken.WaitHandle.WaitOne在任务中的行为的一个小示例,您可以对其进行试验

服务中实现的原始while循环如下所示:

while (!cancellationToken.IsCancellationRequested)
{
    // Processing
    if(connectionReady)
    {
        // Do its business
        break;
    }
    // Pause for 20 seconds before trying again
    Thread.Sleep(20000);
}

在此实现中,线程将被阻塞20秒——不管是否有任何取消触发器。这是潜在的问题——根据与Thread.Sleep调用有关的停止操作的时间,服务管理器将使停止操作超时并强行终止管理服务。

暂停执行代码,但知道取消请求很简单:

while (!cancellationToken.IsCancellationRequested) 
{ 
    // Processing
    if(connectionReady)
    {
        // Do its business
        break;
    }
    // Pause for 20 seconds before trying again - now with cancellation support
    var cancellationTriggered = cancellationToken.WaitHandle.WaitOne(20000);
}

这一行更改实现了与原始实现相同的20秒暂停,并且它也知道任务取消。

兴趣点

如果您发现到处都在使用Thread.Sleep(n),请考虑改为使用CancellationToken.WaitHandle.WaitOne(n)方法。这将有助于保持异常记录的大小。

关注
打赏
1665926880
查看更多评论
立即登录/注册

微信扫码登录

0.0713s