当前位置: 首页 > news >正文

C#中TaskFactory实现线程任务

在 C# 中,TaskFactorySystem.Threading.Tasks命名空间下的一个类,它为任务的创建、管理和调度提供了一个简便的方式。与直接使用Task类相比,TaskFactory提供了更高层次的抽象,允许开发者更加灵活地控制任务的创建、执行、取消、调度等操作。

1.TaskFactory的基本概念

TaskFactory是一个帮助你创建和管理任务的工具类。它通过封装Task的创建过程,使得任务的启动、调度和继续等操作变得更加方便和灵活。

主要功能和用途

  • 任务创建和启动:通过StartNew方法创建并启动新的任务。
  • 任务调度控制:允许通过指定TaskCreationOptionsTaskContinuationOptions来控制任务的行为。
  • 任务的链式操作:支持在一个任务完成后自动执行其他任务。
  • 支持取消任务:可以结合CancellationToken来控制任务的取消。
  • 处理异常:能够在任务异常时进行处理。

2.TaskFactory的基本功能

任务创建与启动 (StartNew)

TaskFactory提供了StartNew方法来创建并启动任务。这是其最常用的功能,它允许你异步执行一个委托(通常是一个ActionFunc)。

1

2

3

4

var factory =newTaskFactory();

Task task = factory.StartNew(() => {

Console.WriteLine("任务开始执行...");

});

在这个例子中,StartNew会启动一个新任务,并异步执行提供的委托。

任务选项控制:TaskCreationOptions

TaskFactory允许你通过TaskCreationOptions来定制任务的创建方式。你可以指定任务的执行方式、优先级等。

常见的选项包括:

  • None:默认选项,任务使用线程池线程执行。
  • LongRunning:表示任务可能需要长时间执行,通常会分配一个新的线程来执行任务,而不是使用线程池。
  • AttachedToParent:指示任务是父任务的一部分。
  • DenyChildAttach:阻止子任务附加到父任务。

1

2

3

4

var factory =newTaskFactory(TaskCreationOptions.LongRunning, TaskContinuationOptions.None);

Task task = factory.StartNew(() => {

Console.WriteLine("长时间运行的任务");

});

任务链式操作:ContinueWith和TaskContinuationOptions

TaskFactory允许你创建任务的链式操作,也就是一个任务完成后可以自动执行另一个任务。你可以使用ContinueWith来设置后续任务,并通过TaskContinuationOptions控制后续任务的执行条件。

例如,你可以让任务只在当前任务成功完成时执行后续任务,或者只在任务失败时执行后续任务。

1

2

3

4

5

6

7

Task task1 = factory.StartNew(() => {

Console.WriteLine("任务1");

});

Task task2 = task1.ContinueWith(t => {

Console.WriteLine("任务2(在任务1完成后执行)");

}, TaskContinuationOptions.OnlyOnRanToCompletion);

任务取消:CancellationToken

TaskFactory可以与CancellationToken配合使用,以便支持任务取消。当你需要取消一个任务时,CancellationTokenSource发出取消请求,Task会检查该请求并在合适的时机停止执行。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

var cts =newCancellationTokenSource();

var factory =newTaskFactory();

Task task = factory.StartNew(() => {

for(inti = 0; i < 10; i++) {

if(cts.Token.IsCancellationRequested) {

Console.WriteLine("任务已取消");

return;

}

Console.WriteLine("任务正在执行...");

Thread.Sleep(1000);

}

}, cts.Token);

// 取消任务

cts.Cancel();

任务异常处理

TaskFactory还支持任务的异常处理。如果任务在执行过程中发生异常,你可以通过ContinueWith捕获并处理这些异常。

1

2

3

4

5

6

7

Task task = factory.StartNew(() => {

thrownewInvalidOperationException("发生错误");

}).ContinueWith(t => {

if(t.Exception !=null) {

Console.WriteLine($"任务发生了异常: {t.Exception.Message}");

}

}, TaskContinuationOptions.OnlyOnFaulted);

在这个例子中,ContinueWith会在任务发生异常时被调用,并处理异常信息。

3.TaskFactory的实现原理与机制

TaskFactory本质上是对Task类的封装,它通过TaskCreationOptionsTaskContinuationOptions提供了更灵活的任务调度和管理方式。

内部工作机制

  • 任务创建和启动TaskFactory提供的方法(如StartNew)会创建一个Task实例,并将任务的执行逻辑传递给Task
  • 任务调度TaskFactory可以通过不同的任务选项来控制任务的调度方式,例如将任务标记为长时间运行的任务。
  • 任务链式操作ContinueWithTaskContinuationOptions使得开发者能够创建任务之间的依赖关系,并定义任务完成后的后续动作。
  • 任务取消:任务的取消是通过CancellationToken实现的。开发者可以通过传递CancellationToken来允许任务在某个条件下提前终止。

4. 主要的使用场景

场景 1:并行任务的启动

TaskFactory用于启动多个并行执行的任务,特别是在需要对任务进行细粒度控制时。例如,在处理多个独立的任务时,你可能会使用TaskFactory来启动这些任务,并通过Task.WhenAll等方法等待所有任务完成。

1

2

3

4

5

6

7

8

9

Task task1 = factory.StartNew(() => {

// 执行任务1

});

Task task2 = factory.StartNew(() => {

// 执行任务2

});

Task.WhenAll(task1, task2).ContinueWith(t => {

Console.WriteLine("所有任务完成");

});

场景 2:长时间运行的任务

使用TaskCreationOptions.LongRunning来创建需要长时间执行的任务。通常,这些任务不应使用线程池线程来执行,而应该单独分配线程,以避免阻塞线程池中的其他任务。

1

2

3

4

var factory =newTaskFactory(TaskCreationOptions.LongRunning, TaskContinuationOptions.None);

Task task = factory.StartNew(() => {

// 执行长时间任务

});

场景 3:任务的依赖关系

当任务之间有依赖关系时,TaskFactoryContinueWith方法非常有用。你可以指定某个任务成功完成后执行另一个任务,或者在任务失败时执行相应的操作。

1

2

3

4

5

6

Task task1 = factory.StartNew(() => {

// 任务1的操作

});

Task task2 = task1.ContinueWith(t => {

// 任务2的操作

}, TaskContinuationOptions.OnlyOnRanToCompletion);

场景 4:任务取消

TaskFactory还适用于需要支持取消操作的任务。例如,在下载文件、处理大批数据等场景中,你可以使用CancellationToken来取消正在执行的任务。


http://www.jsqmd.com/news/876095/

相关文章:

  • Ubuntu 20.04 上为 RTX 3060 编译 OpenCV 4.2.0 + CUDA 时,我踩过的那些坑(附完整解决方案)
  • LLM应用开发之模型微调技术详解
  • 3步轻松解密网易云音乐NCM文件:免费实现音乐跨平台播放
  • NHSE终极指南:动物森友会存档编辑器的5个核心应用场景
  • SketchUp STL插件终极指南:5分钟掌握3D打印模型转换的完整方案
  • GMERF与MERF:处理过离散计数数据的小域估计方法对比
  • JMeter接口测试工业化实践:从脚本编写到CI/CD全链路
  • 茉莉花插件终极指南:如何在3分钟内彻底解决Zotero中文文献管理难题
  • 接口测试三层防御体系:契约校验、逻辑穿透与系统压测
  • Godot 4.3本地AI编程助手:GDScript智能协作者实战指南
  • Edge和Chrome同时罢工?可能是这个Windows服务在搞鬼!附一键排查脚本
  • 3分钟掌握SketchUp STL插件:3D打印模型转换的完整解决方案
  • 终极猫抓浏览器扩展:5个简单步骤轻松捕获在线视频资源的完整指南
  • 高斯随机定时器原理与JMeter压测行为建模
  • JMeter+InfluxDB+Grafana压测监控实时可视化实战
  • TranslucentTB:Windows任务栏透明美化终极指南,轻松打造个性化桌面
  • 第七史诗自动化助手E7Helper:解放双手的游戏效率革命
  • E7Helper:第七史诗自动化助手终极指南,告别重复刷图烦恼
  • 解锁音乐自由:qmcdump如何让被加密的音乐重获新生?
  • 机器学习势函数与连续介质模型在二维材料原子重构中的对比研究
  • 龙蜥8.8系统下,手把手教你安全升级OpenSSH到9.7p1(附防失联指南)
  • 湍流建模不确定性量化:从物理扰动到贝叶斯推断的融合实践
  • 告别Windows文件搜索慢!Listary Pro 6保姆级配置教程,效率翻倍不是梦
  • RTX51任务调度中K_IVL与K_TMO事件详解
  • Zotero文献去重终极指南:一键清理重复条目,专注高效科研
  • Unity找不到ffmpeg.dll的四大根因与实战解决方案
  • 煎饼果仔 夏天妹妹 90 天 AI 变现落地计划
  • KOSS模型:卡尔曼滤波与深度学习的融合创新
  • AutoML与集成学习在多模态医疗AI中的工程化实践
  • 数据缺失处理与PCA降维:构建全球生活便利指数的技术实践