实现 .NET 4.0 下的 Task 类相似功能:TaskExCum 组件详解

引言

随着 .NET 技术的发展,异步编程模型逐渐成为现代应用程序开发中的标准实践之一。.NET 4.5 引入了 Task 类,极大地简化了异步编程的过程。然而,许多遗留系统仍在使用 .NET 4.0 或更低版本,这些版本并未直接支持 Task 类的全部功能。为此,我们开发了 TaskExCum 组件,旨在为 .NET 4.0 提供与 .NET 4.5 相似的 Task 功能,包括 Task.Run()Task.WhenAll() 方法。

组件概述

TaskExCum 是一个静态类,提供了以下主要功能:

  • Run 方法:用于异步执行任务,并获取任务的结果。
  • WhenAll 方法:用于等待多个任务完成,并收集所有任务的结果。

实现步骤

接下来,我们将详细讲解 TaskExCum 组件的实现步骤,以便读者能够更好地理解其工作原理,并将其应用于自己的项目中。

步骤 1: 创建 TaskExCum

首先,我们需要创建一个静态类 TaskExCum,并在其中定义静态方法。

public static class TaskExCum
{
// 方法定义将在后续步骤中添加
}

步骤 2: 实现 Run 方法

Run 方法允许开发者异步执行任务,并获取任务的结果。我们为 Run 方法提供了两种重载形式,分别用于执行无返回值的操作(Action)和有返回值的操作(Func<TResult>)。

public static Task<TResult> Run<TResult>(Func<TResult> function)
{
#if NET45
// 如果目标框架是 .NET 4.5 或更高版本,使用 Task.Run 方法
return Task.Run(function);
#else
// 如果目标框架是 .NET 4.0,使用 Task.Factory.StartNew 方法
return Task.Factory.StartNew(
function,
CancellationToken.None,
TaskCreationOptions.None,
TaskScheduler.Default
);
#endif
} public static Task Run(Action action)
{
#if NET45
// 如果目标框架是 .NET 4.5 或更高版本,使用 Task.Run 方法
return Task.Run(action);
#else
// 如果目标框架是 .NET 4.0,使用 Task.Factory.StartNew 方法
return Task.Factory.StartNew(
action,
CancellationToken.None,
TaskCreationOptions.None,
TaskScheduler.Default
);
#endif
}

详细解释

  • 条件编译:使用 #if NET45 编译符号,当项目目标框架为 .NET 4.5 及更高版本时,使用 Task.Run 方法。
  • Task.Factory.StartNew:当项目目标框架为 .NET 4.0 时,使用 Task.Factory.StartNew 方法来启动任务。
    • CancellationToken.None:表示没有取消令牌,任务不会被外部取消。
    • TaskCreationOptions.None:表示没有特殊的任务创建选项。
    • TaskScheduler.Default:这是关键点之一。TaskScheduler.Default 表示使用默认的线程池调度器,这意味着任务会在线程池中的一个线程上执行,而不是每次都启动一个新的线程。这有助于提高性能和资源利用率。

步骤 3: 实现 WhenAll 方法

WhenAll 方法用于等待多个任务完成,并收集所有任务的结果。我们为 WhenAll 方法提供了多种重载形式,以支持不同类型的任务集合。

public static Task<TResult[]> WhenAll<TResult>(IEnumerable<Task<TResult>> tasks)
{
#if NET45
// 如果目标框架是 .NET 4.5 或更高版本,使用 Task.WhenAll 方法
return Task.WhenAll(tasks);
#else
// 如果目标框架是 .NET 4.0,调用 WhenAllCore 方法
return WhenAllCore(tasks);
#endif
} public static Task<TResult[]> WhenAll<TResult>(params Task<TResult>[] tasks)
{
#if NET45
// 如果目标框架是 .NET 4.5 或更高版本,使用 Task.WhenAll 方法
return Task.WhenAll(tasks);
#else
// 如果目标框架是 .NET 4.0,调用 WhenAllCore 方法
return WhenAllCore(tasks);
#endif
} public static Task WhenAll(IEnumerable<Task> tasks)
{
#if NET45
// 如果目标框架是 .NET 4.5 或更高版本,使用 Task.WhenAll 方法
return Task.WhenAll(tasks);
#else
// 如果目标框架是 .NET 4.0,调用 WhenAllCore 方法
return WhenAllCore(tasks);
#endif
}

详细解释

  • 条件编译:使用 #if NET45 编译符号,当项目目标框架为 .NET 4.5 及更高版本时,使用 Task.WhenAll 方法。
  • WhenAllCore:当项目目标框架为 .NET 4.0 时,调用 WhenAllCore 方法来实现相同的功能。

步骤 4: 实现 WhenAllCore 方法

WhenAllCore 方法是 WhenAll 方法的核心实现,负责处理任务集合,等待所有任务完成,并收集结果或异常信息。

private static Task WhenAllCore(IEnumerable<Task> tasks)
{
return WhenAllCore(tasks, (completedTasks, tcs) => tcs.TrySetResult(null));
} private static Task<TResult[]> WhenAllCore<TResult>(IEnumerable<Task<TResult>> tasks)
{
return WhenAllCore(tasks.Cast<Task>(), (completedTasks, tcs) =>
{
tcs.TrySetResult(completedTasks.Select(t => ((Task<TResult>)t).Result).ToArray());
});
} private static Task<TResult> WhenAllCore<TResult>(IEnumerable<Task> tasks, Action<Task[], TaskCompletionSource<TResult>> setResultAction)
{
if (tasks == null)
{
throw new ArgumentNullException("tasks");
} Contract.EndContractBlock();
Contract.Assert(setResultAction != null); var tcs = new TaskCompletionSource<TResult>();
var array = (tasks as Task[]) ?? tasks.ToArray(); if (array.Length == 0)
{
// 如果任务集合为空,直接设置结果
setResultAction(array, tcs);
}
else
{
Task.Factory.ContinueWhenAll(array, completedTasks =>
{
var exceptions = new List<Exception>();
bool hasCanceled = false; foreach (var task in completedTasks)
{
if (task.IsFaulted)
{
// 收集所有失败任务的异常信息
exceptions.AddRange(task.Exception.InnerExceptions);
}
else if (task.IsCanceled)
{
// 检查是否有任务被取消
hasCanceled = true;
}
} if (exceptions.Count > 0)
{
// 如果有异常,设置异常结果
tcs.TrySetException(exceptions);
}
else if (hasCanceled)
{
// 如果有任务被取消,设置取消结果
tcs.TrySetCanceled();
}
else
{
// 如果没有异常且没有任务被取消,设置成功结果
setResultAction(completedTasks, tcs);
}
}, CancellationToken.None, TaskContinuationOptions.ExecuteSynchronously, TaskScheduler.Default);
} return tcs.Task;
}

详细解释

  • 参数验证:检查传入的任务集合是否为 null,如果是,则抛出 ArgumentNullException
  • TaskCompletionSource:创建一个 TaskCompletionSource 对象,用于管理任务的完成状态。
  • 任务转换:将任务集合转换为数组,以便于后续处理。
  • 任务数量检查:如果任务集合为空,直接调用 setResultAction 设置结果。
  • 等待所有任务完成:使用 Task.Factory.ContinueWhenAll 方法等待所有任务完成。
    • 异常处理:遍历已完成的任务,收集所有失败任务的异常信息。
    • 取消处理:检查是否有任务被取消。
    • 设置结果:如果没有异常且没有任务被取消,调用 setResultAction 设置结果。
    • TaskScheduler.Default:这里再次使用 TaskScheduler.Default,确保任务在默认的线程池中执行,而不是每次都启动新的线程。

步骤 5: 添加异常处理逻辑

为了确保组件的健壮性,我们还需要在 WhenAllCore 方法中添加异常处理逻辑,确保所有异常都能被捕获并正确处理。

private static void AddPotentiallyUnwrappedExceptions(ref List<Exception> targetList, Exception exception)
{
var ex = exception as AggregateException;
Contract.Assert(exception != null);
Contract.Assert(ex == null || ex.InnerExceptions.Count > 0); if (targetList == null)
{
targetList = new List<Exception>();
} if (ex != null)
{
// 如果异常是 AggregateException,添加其内部异常
targetList.Add(ex.InnerExceptions.Count == 1 ? ex.InnerExceptions[0] : ex);
}
else
{
// 否则,直接添加异常
targetList.Add(exception);
}
}

详细解释

  • 异常类型检查:检查传入的异常是否为 AggregateException
  • 异常列表初始化:如果 targetListnull,则初始化一个新的列表。
  • 异常添加:根据异常的类型,将异常或其内部异常添加到列表中。

示例代码

为了帮助读者更好地理解如何使用 TaskExCum 组件,下面是一些示例代码。

示例 1: 使用 Run 方法

using System;
using System.Threading.Tasks; class Program
{
static void Main(string[] args)
{
try
{
// 异步执行任务并等待结果
string result = TaskExCum.Run(() => "Hello from Task!").Result;
Console.WriteLine(result);
}
catch (Exception ex)
{
Console.WriteLine($"Error: {ex.Message}");
}
}
}

示例 2: 使用 WhenAll 方法

using System;
using System.Linq;
using System.Threading.Tasks; class Program
{
static void Main(string[] args)
{
try
{
// 创建多个任务
var tasks = Enumerable.Range(1, 5).Select(i => TaskExCum.Run(() => i * i)).ToArray(); // 等待所有任务完成并获取结果
int[] results = TaskExCum.WhenAll(tasks).Result; // 输出结果
foreach (var result in results)
{
Console.WriteLine(result);
}
}
catch (Exception ex)
{
Console.WriteLine($"Error: {ex.Message}");
}
}
}

结论

通过 TaskExCum 组件,即使是在 .NET 4.0 这样的老框架版本中,我们也能够享受到现代异步编程模型带来的便利。希望这个组件能够帮助那些需要在旧版 .NET 框架中实现异步操作的开发者们,提高他们的开发效率和代码质量。如果你有任何建议或改进意见,欢迎留言交流!


https://i.cnblogs.com/posts/edit;postId=18515670

以上就是关于 TaskExCum 组件的详细介绍。希望通过这篇文章,读者能够更好地理解和使用这个组件,从而在自己的项目中实现高效的异步编程。如果有任何问题或需要进一步的帮助,请随时留言!

实现.NET 4.0下的Task类相似功能组件的更多相关文章

  1. C# 语言规范_版本5.0 (第10章 类)

    1. 类 类是一种数据结构,它可以包含数据成员(常量和字段).函数成员(方法.属性.事件.索引器.运算符.实例构造函数.静态构造函数和析构函数)以及嵌套类型.类类型支持继承,继承是一种机制,它使派生类 ...

  2. .net core2.0下使用Identity改用dapper存储数据

    前言. 已经好多天没写博客了,鉴于空闲无聊之时又兴起想写写博客,也当是给自己做个笔记.过了这么些天,我的文笔还是依然那么烂就请多多谅解了.今天主要是分享一下在使用.net core2.0下的实际遇到的 ...

  3. C#6.0语言规范(十) 类

    类是可以包含数据成员(常量和字段),函数成员(方法,属性,事件,索引器,运算符,实例构造函数,析构函数和静态构造函数)和嵌套类型的数据结构.类类型支持继承,这是一种派生类可以扩展和专门化基类的机制. ...

  4. 在 .NET 4.0 下编写扩展代码以支持 async 异步编程

    微软在C# 5中引入了async.await这两个异步编程的关键字,要使用这两个关键字需要你的IDE支持C#5.0语法,也就意味着你需要使用VS 2012版本以上IDE,或者在Vs2010卸载其编译器 ...

  5. Foundation框架下的常用类:NSNumber、NSDate、NSCalendar、NSDateFormatter、NSNull、NSKeyedArchiver

    ========================== Foundation框架下的常用类 ========================== 一.[NSNumber] [注]像int.float.c ...

  6. hdwiki model目录下的函数类

    model目录下的函数类    actions.class.php(站内地图相关) getHTML:获得页面菜单和相关信息 getMap:生成站内地图 adv.class.php 对wiki_adve ...

  7. .NET Framework4.0 下的多线程

    一.简介 在4.0之前,多线程只能用Thread或者ThreadPool,而4.0下提供了功能强大的Task处理方式,这样免去了程序员自己维护线程池,而且可以申请取消线程等...所以本文主要描述Tas ...

  8. Darwin Streaming server 的 Task 类

    Darwin Streaming Server 是一个开放源代码的streaming server,对于streaming server的编程和软件结构有着一定的参考价值,它是使用C++写的,其中的并 ...

  9. Task类(任务)

    任务表示应完成的某个单元的工作.这个单元的工作可以在单独的线程中运行,也可以以同步方式启动一个任务,这需要等待主调用线程.使用任务不仅可以获得一个抽象层,还可以对底层线程进行很多控制. 1.启动任务 ...

  10. 在VC6.0下如何调用Delphi5.0开发的进程内COM

    因为本人的语言水平很差,考大学时150的总分,我考了个60分.外语也是,初中及格过一次,会考及格过一次.其它的时间好像从没有及格过.所以我不写文章,因我一百字的文章给我写,至少要出八九个错别字.哈哈… ...

随机推荐

  1. 新晋 Committer!来自复旦大学的帅哥一枚

    点亮Star️ · 支持我们 https://github.com/apache/dolphinscheduler 最近,社区星力量又迎来一位新晋 Committer,这次是来自复旦大学研究生在读的王 ...

  2. Avnet ZUBoard 1CG开发板上手—深度学习新选择

    Avnet ZUBoard 1CG 开发板上手-深度学习新选择 摘要 本文主要介绍了 Avnet ZUBoard 1CG 开发板的特性.架构.硬件单元等概念,并对如何使用以太网接口和串口连接开发板进行 ...

  3. Mac M1 安装Homebrew

    /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)"

  4. 使用 refreshNuxtData 刷新 Nuxt应用 中的数据

    title: 使用 refreshNuxtData 刷新 Nuxt应用 中的数据 date: 2024/8/21 updated: 2024/8/21 author: cmdragon excerpt ...

  5. 讲讲Java的序列化反序列化?

    序列化:把对象转换为字节序列的过程称为对象的序列化. 反序列化:把字节序列恢复为对象的过程称为对象的反序列化. 什么时候会用到 当只在本地 JVM 里运行下 Java 实例,这个时候是不需要什么序列化 ...

  6. echarts x轴下绘制表

    效果图: 把下面代码复制到官网实例的js代码编辑中即可预览( 附连接:Examples - Apache ECharts) let map = { 销售单价: [2200.0,4000.9,700.0 ...

  7. Windows 设置 FRP 自动启动

    由于 frps/frpc 不是 Windows 服务应用程序,因此我们不能直接使用 New-Service 命令创建 frps/frpc 服务.我们可以使用下面的方法将 frps/frpc 封装为 W ...

  8. 编译器实现之旅——第十六章 代码装载、链接器、全局变量与main函数

    在上一章的旅程中,我们已经实现了函数调用的代码生成器分派函数,但在上一章的末尾,我们留下了三个问题: 我们需要为全局变量压栈 main函数需要在程序启动时被自动调用 我们需要实现一个链接器,以将所有的 ...

  9. maven 打包 pom build

    <dependencyManagement> <dependencies> <dependency> <groupId>org.springframew ...

  10. 录音转文字SDK哪家强?

    最近在做一款录音App,有一个模块是录音转文字功能,于是对比了市面上常见的API,国内做的比较大的主要有讯飞.腾讯.阿里.百度.华为. 讯飞 讯飞在国内做语音SDK是做的比较早的,翻译出来的准确率挺不 ...