首页 > 编程语言 >这样在 C# 使用 LongRunnigTask 是错的

这样在 C# 使用 LongRunnigTask 是错的

时间:2023-03-06 09:02:06浏览次数:62  
标签:Status Task C# TaskCreationOptions ThreadPool LongRunning 使用 LongRunnigTask 线程

Task.Factory.StartNew 有一个重载,是支持 TaskCreationOptions.LongRunning 参数来指定 Task 的特征的。但是可能在没有注意的情况下,你就使用了错误的用法。那么本文我们来简单阐述一下这个参数的作用,和使用的注意要点。

这样其实是错误的

有的时候,你可能会这么写:

Task.Factory.StartNew(async () =>
{
    while (true)
    {
        // do something
        await Task.Delay(1000);
    }
}, TaskCreationOptions.LongRunning);

但其实,这是个错误的写法。

为什么需要 LongRunning

我们通常两种情况下会想到使用 TaskCreationOptions.LongRunning 参数:

  1. 你的任务需要长时间运行,比如一个循环,或者一个死循环。用来从队列中取数据,然后处理数据,或者是一些定时任务。
  2. 你的任务需要占用大量的 CPU 资源,是一个很大的循环,比如要遍历一个很大的数组,并做一些处理。

那么这个时候,我们就需要使用 TaskCreationOptions.LongRunning 参数来指定 Task。

因为我们可能学习到了,Task 默认的 Scheduler 是 ThreadPool,而 ThreadPool 的线程是有限的,如果你的任务需要长时间运行,或者是需要占用大量的 CPU 资源,那么就会导致 ThreadPool 的线程不够用。导致线程饥饿,或者是线程池的线程被占用,导致其他的任务无法执行。

于是我们很聪明的就想到了,我们可以使用 TaskCreationOptions.LongRunning 参数来指定 Task,这样就可以避免线程饥饿。

弄巧成拙

但是实际上,开篇的写法并不能达到我们的目的。

我们可以通过以下代码来验证一下:

var task = Task.Factory.StartNew(async () =>
{
    while (true)
    {
        // do something
        await Task.Delay(1000);
    }
}, TaskCreationOptions.LongRunning);

Thread.Sleep(3000);

Console.WriteLine($"Task Status: {task.Status}");
// Task Status: RanToCompletion

我们可以看到,Task 的状态是并非是 Running,而是 RanToCompletion。

也就是说,我们的任务在 3 秒后就已经执行完了,而不是我们想要的长时间运行。

究其原因,是因为我们采用了异步的方式来执行任务。而异步任务的执行,是通过 ThreadPool 来执行的。也就是说,虽然我们使用了 TaskCreationOptions.LongRunning 参数,来想办法指定线程池单独开一个线程,但是实际上在一个 await 之后,我们的任务还是在 ThreadPool 中执行的。

这会导致,我们的任务实际上后续又回到了 ThreadPool 中,而不是我们想要的单独的线程。起不到单独长期运行的作用。

正确的写法

因此,实际上如果想要保持单独的线程持续的运行,我们需要移除异步的方式,改为同步的方式。

var task = Task.Factory.StartNew(() =>
{
    while (true)
    {
        // do something
        Thread.Sleep(1000);
    }
}, TaskCreationOptions.LongRunning);

Thread.Sleep(3000);

Console.WriteLine($"Task Status: {task.Status}");
// Task Status: Running

这样我们就可以看到,Task 的状态是 Running,而不是 RanToCompletion。我们通过 TaskCreationOptions.LongRunning 参数,单独开启的线程就可以一直运行下去。

实际上还有很多考量

要考量 TaskScheduler 的实现

本文采用的是 aspnetcore 的实现,但是在其他的实现中,可能会有不同的实现。你也完全有可能实现一个 await 之后,不回到 ThreadPool 的实现。

LongRunning 也不是就不能用异步

正如开篇提到的第二种场景,如果你的业务是在第一个 await 之前有大量的同步代码,那么此时单独开启一个线程,也是有意义的。

我就是一个死循环,里面也是异步的怎么办

那么你可以考虑让这个 LongRuning 的 Task,不要 await,而是通过 Wait() 来等待。这样就可以避免 LongRunning 的 Task 直接结束。

总结

本文我们简单阐述了 TaskCreationOptions.LongRunning 参数的作用,和使用的注意要点。

参考

感谢阅读,如果觉得本文有用,不妨点击推荐

标签:Status,Task,C#,TaskCreationOptions,ThreadPool,LongRunning,使用,LongRunnigTask,线程
From: https://www.cnblogs.com/newbe36524/p/0x026-This-is-the-wrong-way-to-use-LongRunnigTask-in-

相关文章

  • ElasticSearch 实现分词全文检索 - Restful基本操作
    Restful语法GET请求:http://ip:port/index:查询索引信息http://ip;port/index/type/doc_id:查询指定的文档信息POST请求:http://ip;port/index/type/_search:......
  • docker-compose搭建redis哨兵模式(一主二从三哨兵)
    参考:https://blog.csdn.net/band_mmbx/article/details/1264280411、安装docker和docker-compose2、配置redis2.1下载redis.conf配置文件因为docker启动redis是默认没......
  • 用MiniPC搭建个人服务器
    最近突然对小型电子产品产生的兴趣,经过一段时间调查,最终选择从迷你PC下手。因为类似树莓派的产品,还是有一定的上手门槛的。开发板类的产品也更偏硬件一点。而迷你PC除了......
  • git 日常使用
    原理部分代码版本控制发展历程手动复制,命令不同不版本,缺点:容易出错。CVS,集中式代码管理版本控制系统,缺点:单点故障。DVCS,分布式代码管理版本控制系统。工作原理在......
  • MCU并行协议
    @目录简介8080并口协议6800并口协议简介本文将MOTOROLA的M6800和INTERL的I8080总线协议统称为MCU并行协议。因为其在高速,近距离接口在数据传输方面的优越性,被广泛应用于......
  • 轻量级压测平台RunnerGo简介及使用教程
    RunnerGo是一个功能强大,使用简单的性能测试平台,它基于go语言开发,支持接口管理、自动化测试、性能测试等功能。更重要的是,RunnerGo完全开源。下图为RunnerGo首页的数据大屏......
  • BeyondCompare
    @目录前言下载安装问题超过试用期解决办法bat脚本自动处理试用期办法前言BeyondCompare是一套由ScooterSoftware推出的文件比较工具。主要用途是对比两个文件夹或者文......
  • electron解包与压缩
    安装asarnpminstallasar-g解包asarextract<filename>.asar<filename>进入<fileanme>文件夹asarp./<filename>.asar将生成的<filename>.asar把原来的.asa......
  • How to learn Aptos Core?
    TolearnAptosCore,youcanfollowthesesteps:1.Familiarizeyourselfwiththebasicsofcomputerprogrammingandsoftwaredevelopment.2.Learnaboutthef......
  • Vue过滤器的使用详解(代码实现)
    过滤器的功能是对要显示的数据进行格式化后再显示,其并没有改变原本的数据,只是产生新的对应的数据 过滤器,其实不是必须要用的东西,它只是vue给我们提供的新的数据处理方式......