热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

TPL实现Task.WhileAll扩展方法

文章翻译整理自NikolaMalovic两篇博文:Task.WhileAllAwaitabletaskprogressreporting当Task.WhenAll遇见

文章翻译整理自 Nikola Malovic 两篇博文:

  • Task.WhileAll
  • Awaitable task progress reporting

当 Task.WhenAll 遇见 Task.WhenAny

在 TPL (Task Parallel Library) 中,有两种通过非阻塞方式等待 Task 数组任务结束的方式:Task.WhenAll 和 Task.WhenAny 。

它们的工作方式是:

  • WhenAll 当每项任务都完成时为完成。
  • WhenAny 当任意项任务完成时为完成。

现在我们需要一项功能,完成 Task 数组中的所有任务,并且当有任务完成时汇报状态。

我们称这个扩展方法为:Task.WhileAll 。

扩展方法实现

1 public static class TaskExtensions
2 {
3 public static async Task> WhileAll(this IList> tasks, IProgress progress)
4 {
5 var result = new List(tasks.Count);
6 var done = new List>(tasks);
7
8 while (done.Count > 0)
9 {
10 await Task.WhenAny(tasks);
11
12 var spinning = new List>(done.Count - 1);
13 for (int i = 0; i )
14 {
15 if (done[i].IsCompleted)
16 {
17 result.Add(done[i].Result);
18 progress.Report(done[i].Result);
19 }
20 else
21 {
22 spinning.Add(done[i]);
23 }
24 }
25
26 done = spinning;
27 }
28
29 return result;
30 }
31 }

代码实现很简单:

  • 其是 IList> 的一个 async 扩展方法
  • 方法返回完整的 IList 结果
  • 方法会接受一个 IProgress 类型的参数,用于向订阅者发布 Task 完成信息
  • 在方法体内,我们使用一个循环来检测,直到所有 Task 完成
  • 通过使用 Task.WhenAny 来异步等待 Task 完成

单元测试

1 [TestClass]
2 public class UnitTest1
3 {
4 [TestMethod]
5 public async Task TestTaskExtensionsWhileAll()
6 {
7 var task1 = Task.Run(() => 101);
8 var task2 = Task.Run(() => 102);
9 var tasks = new Listint>>() { task1, task2 };
10
11 List<int> result &#61; new List<int>();
12 var listener &#61; new Progress<int>(
13 taskResult &#61;>
14 {
15 result.Add(taskResult);
16 });
17
18 var actual &#61; await tasks.WhileAll(listener);
19 Thread.Sleep(50); // wait a bit for progress reports to complete
20
21 Assert.AreEqual(2, result.Count);
22 Assert.IsTrue(result.Contains(101));
23 Assert.IsTrue(result.Contains(102));
24
25 Assert.AreEqual(2, actual.Count);
26 Assert.IsTrue(actual.Contains(101));
27 Assert.IsTrue(actual.Contains(102));
28 }
29 }

同样&#xff0c;测试代码也不复杂&#xff1a;

  • 创建两个哑元 Task&#xff0c;并存到数组中
  • 定义进度侦听器 Progress&#xff0c;来监测每个任务运行的结果
  • 通过 await 方式来调用方法
  • 使用 Thread.Sleep 来等待 50ms &#xff0c;以便 Progress 可以来得及处理结果
  • 检查所有 Task 执行完毕后均已上报 Progress
  • 检查所有 Task 均已执行完毕

我知道每当使用 Thread.Sleep 时绝不是件好事&#xff0c;所以我决定摆脱它。

实现IProgressAsync

问题实际上是因为 IProgress 接口定义的是 void 委托&#xff0c;因此无法使用 await 进行等待。

因此我决定定义一个新的接口&#xff0c;使用同样的 Report 行为&#xff0c;但会返回 Task &#xff0c;用以实现真正的异步。

1 public interface IProgressAsync<in T>
2 {
3 Task ReportAsync(T value);
4 }

有了异步版本的支持&#xff0c;将使订阅者更容易处理 await 调用。当然也可以使用 async void 来达成&#xff0c;但我认为 async void 总会延伸出更差的设计。所以&#xff0c;我还是选择通过定义 Task 返回值签名的接口来达成这一功能。

如下为接口实现&#xff1a;

1 public class ProgressAsync : IProgressAsync
2 {
3 private readonly Func handler;
4
5 public ProgressAsync(Func handler)
6 {
7 this.handler &#61; handler;
8 }
9
10 public async Task ReportAsync(T value)
11 {
12 await this.handler.InvokeAsync(value);
13 }
14 }

显然也没什么特别的&#xff1a;

  • 使用 Func 来代替 Action&#xff0c;以便可以使用 await
  • ReportAsync 通过使用 await 方式来提供 Task

有了这些之后&#xff0c;我们来更新扩展方法&#xff1a;

1 public static class TaskExtensions
2 {
3 public static async Task> WhileAll(this IList> tasks, IProgressAsync progress)
4 {
5 var result &#61; new List(tasks.Count);
6 var remainingTasks &#61; new List>(tasks);
7
8 while (remainingTasks.Count > 0)
9 {
10 await Task.WhenAny(tasks);
11 var stillRemainingTasks &#61; new List>(remainingTasks.Count - 1);
12 for (int i &#61; 0; i )
13 {
14 if (remainingTasks[i].IsCompleted)
15 {
16 result.Add(remainingTasks[i].Result);
17 await progress.ReportAsync(remainingTasks[i].Result);
18 }
19 else
20 {
21 stillRemainingTasks.Add(remainingTasks[i]);
22 }
23 }
24
25 remainingTasks &#61; stillRemainingTasks;
26 }
27
28 return result;
29 }
30
31 public static Task InvokeAsync(this Func task, T value)
32 {
33 return Task.Factory.FromAsync(task.BeginInvoke, task.EndInvoke, value, null);
34 }
35 }

所有都就绪后&#xff0c;我们就可以将 Thread.Sleep 从单元测试中移除了。

1 [TestClass]
2 public class UnitTest1
3 {
4 private List<int> result &#61; new List<int>();
5 private async Task OnProgressAsync(int arg)
6 {
7 result.Add(arg);
8 }
9
10 [TestMethod]
11 public async Task TestTaskExtensionsWhileAll()
12 {
13 var task1 &#61; Task.Run(() &#61;> 101);
14 var task2 &#61; Task.Run(() &#61;> 102);
15 var tasks &#61; new Listint>>() { task1, task2 };
16
17 var listener &#61; new ProgressAsync<int>(this.OnProgressAsync);
18 var actual &#61; await tasks.WhileAll(listener);
19
20 Assert.AreEqual(2, this.result.Count);
21 Assert.IsTrue(this.result.Contains(101));
22 Assert.IsTrue(this.result.Contains(102));
23
24 Assert.AreEqual(2, actual.Count);
25 Assert.IsTrue(actual.Contains(101));
26 Assert.IsTrue(actual.Contains(102));
27 }
28 }






本文转自匠心十年博客园博客&#xff0c;原文链接&#xff1a;http://www.cnblogs.com/gaochundong/p/tpl_task_whileall.html&#xff0c;如需转载请自行联系原作者



推荐阅读
  • 投融资周报 | Circle 达成 4 亿美元融资协议,唯一艺术平台 A 轮融资超千万美元 ... [详细]
  • 在腾讯云服务器上部署Nginx的详细指南中,首先需要确保安装必要的依赖包。如果这些依赖包已安装,可直接跳过此步骤。具体命令包括 `yum -y install gcc gcc-c++ wget net-tools pcre-devel zlib-devel`。接下来,本文将详细介绍如何下载、编译和配置Nginx,以确保其在腾讯云服务器上顺利运行。此外,还将提供一些优化建议,帮助用户提升Nginx的性能和安全性。 ... [详细]
  • 本文介绍了UUID(通用唯一标识符)的概念及其在JavaScript中生成Java兼容UUID的代码实现与优化技巧。UUID是一个128位的唯一标识符,广泛应用于分布式系统中以确保唯一性。文章详细探讨了如何利用JavaScript生成符合Java标准的UUID,并提供了多种优化方法,以提高生成效率和兼容性。 ... [详细]
  • 在 Kubernetes 中,Pod 的调度通常由集群的自动调度策略决定,这些策略主要关注资源充足性和负载均衡。然而,在某些场景下,用户可能需要更精细地控制 Pod 的调度行为,例如将特定的服务(如 GitLab)部署到特定节点上,以提高性能或满足特定需求。本文深入解析了 Kubernetes 的亲和性调度机制,并探讨了多种优化策略,帮助用户实现更高效、更灵活的资源管理。 ... [详细]
  • SQLite数据库CRUD操作实例分析与应用
    本文通过分析和实例演示了SQLite数据库中的CRUD(创建、读取、更新和删除)操作,详细介绍了如何在Java环境中使用Person实体类进行数据库操作。文章首先阐述了SQLite数据库的基本概念及其在移动应用开发中的重要性,然后通过具体的代码示例,逐步展示了如何实现对Person实体类的增删改查功能。此外,还讨论了常见错误及其解决方法,为开发者提供了实用的参考和指导。 ... [详细]
  • 探索偶数次幂二项式系数的求和方法及其数学意义 ... [详细]
  • HBase Java API 进阶:过滤器详解与应用实例
    本文详细探讨了HBase 1.2.6版本中Java API的高级应用,重点介绍了过滤器的使用方法和实际案例。首先,文章对几种常见的HBase过滤器进行了概述,包括列前缀过滤器(ColumnPrefixFilter)和时间戳过滤器(TimestampsFilter)。此外,还详细讲解了分页过滤器(PageFilter)的实现原理及其在大数据查询中的应用场景。通过具体的代码示例,读者可以更好地理解和掌握这些过滤器的使用技巧,从而提高数据处理的效率和灵活性。 ... [详细]
  • 本文探讨了Android系统中支持的图像格式及其在不同版本中的兼容性问题,重点涵盖了存储、HTTP传输、相机功能以及SparseArray的应用。文章详细分析了从Android 10 (API 29) 到Android 11 的存储规范变化,并讨论了这些变化对图像处理的影响。此外,还介绍了如何通过系统升级和代码优化来解决版本兼容性问题,以确保应用程序在不同Android版本中稳定运行。 ... [详细]
  • FastDFS Nginx 扩展模块的源代码解析与技术剖析
    FastDFS Nginx 扩展模块的源代码解析与技术剖析 ... [详细]
  • 本文深入探讨了CGLIB BeanCopier在Bean对象复制中的应用及其优化技巧。相较于Spring的BeanUtils和Apache的BeanUtils,CGLIB BeanCopier在性能上具有显著优势。通过详细分析其内部机制和使用场景,本文提供了多种优化方法,帮助开发者在实际项目中更高效地利用这一工具。此外,文章还讨论了CGLIB BeanCopier在复杂对象结构和大规模数据处理中的表现,为读者提供了实用的参考和建议。 ... [详细]
  • 动态壁纸 LiveWallPaper:让您的桌面栩栩如生(第二篇)
    在本文中,我们将继续探讨如何开发动态壁纸 LiveWallPaper,使您的桌面更加生动有趣。作为 2010 年 Google 暑期大学生博客分享大赛 Android 篇的一部分,我们将详细介绍 Ed Burnette 的《Hello, Android》第三版中的相关内容,并分享一些实用的开发技巧和经验。通过本篇文章,您将了解到如何利用 Android SDK 创建引人入胜的动态壁纸,提升用户体验。 ... [详细]
  • 能够感知你情绪状态的智能机器人即将问世 | 科技前沿观察
    本周科技前沿报道了多项重要进展,包括美国多所高校在机器人技术和自动驾驶领域的最新研究成果,以及硅谷大型企业在智能硬件和深度学习技术上的突破性进展。特别值得一提的是,一款能够感知用户情绪状态的智能机器人即将问世,为未来的人机交互带来了全新的可能性。 ... [详细]
  • 在IIS上运行的WebApi应用程序在开发环境中能够正常进行文件的读写操作。然而,在尝试通过FTP访问实时服务器上的文件列表时,遇到了无法显示的问题,尽管服务器配置与开发环境相同。这可能涉及权限设置、FTP服务配置或网络连接等方面的问题。 ... [详细]
  • 本文详细介绍了在Windows XP系统中安装和配置Unix打印服务的方法,以支持远程行式打印机(LPR)功能。对于同时使用Windows 2000 Server打印服务器和Unix打印服务器的网络环境,该指南提供了实用的步骤和配置建议,确保不同平台之间的兼容性和高效打印。 ... [详细]
  • 本文介绍了如何利用Apache POI库高效读取Excel文件中的数据。通过实际测试,除了分数被转换为小数存储外,其他数据均能正确读取。若在使用过程中发现任何问题,请及时留言反馈,以便我们进行更新和改进。 ... [详细]
author-avatar
mobiledu2502860057
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有