iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >C#怎么使用Task实现执行并行任务
  • 655
分享到

C#怎么使用Task实现执行并行任务

2023-07-05 21:07:08 655人浏览 独家记忆
摘要

这篇“C#怎么使用Task实现执行并行任务”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C#怎么使用Task实现执行并行任务

这篇“C#怎么使用Task实现执行并行任务”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C#怎么使用Task实现执行并行任务”文章吧。

    一、Task执行并行任务的原理

    使用Task执行并行任务的原理是将任务分成多个小块,每个小块都可以在不同的线程上运行。然后,使用Task.Run方法将这些小块作为不同的任务提交给线程池。线程池会自动管理线程的创建和销毁,并根据系统资源的可用情况来自动调整线程数量,从而实现最大化利用CPU资源的效果。

    二、5个示例展示

    示例1

    下面是一个简单的示例,展示如何使用Task来执行并行任务:

    void Task1(){    // 创建任务数组    var tasks = new Task[10];    for (var i = 0; i < tasks.Length; i++)    {        var taskId = i + 1;        // 使用Task.Run方法提交任务        tasks[i] = Task.Run(() =>        {            Console.WriteLine("任务 {0} 运行在线程 {1} 中", taskId, Task.CurrentId);            // 执行任务逻辑        });    }    // 等待所有任务完成    Task.WaitAll(tasks);    Console.WriteLine("所有任务运行完成。");    Console.ReadKey();}

    在这个示例中,我们创建了一个长度为10的任务数组,然后使用Task.Run方法将每个任务提交给线程池。在任务执行时,使用Task.CurrentId属性获取当前任务的ID,并打印出来以方便观察。最后,我们使用Task.WaitAll方法等待所有任务完成并打印出一条完成信息。

    运行的结果:

    任务 3 运行在线程 11 中
    任务 4 运行在线程 12 中
    任务 8 运行在线程 16 中
    任务 1 运行在线程 9 中
    任务 2 运行在线程 10 中
    任务 5 运行在线程 13 中
    任务 6 运行在线程 14 中
    任务 7 运行在线程 15 中
    任务 9 运行在线程 17 中
    任务 10 运行在线程 18 中
    所有任务运行完成。

    值得注意的是,在实际开发中,需要根据具体情况来评估任务的大小和数量,以确保并行任务的效率和可靠性。

    示例2

    另一个使用Task的示例是计算斐波那契数列。我们可以将斐波那契数列的每一项看成一个任务,然后使用Task.WaitAll方法等待所有任务完成。

    void Task2(){    static long Fib(int n)    {        if (n is 0 or 1)        {            return n;        }        else        {            return Fib(n - 1) + Fib(n - 2);        }    }    const int n = 10; // 计算斐波那契数列的前n项    var tasks = new Task<long>[n];    for (var i = 0; i < n; i++)    {        var index = i; // 需要在闭包内使用循环变量时需要赋值给另外一个变量        if (i < 2)        {            tasks[i] = Task.FromResult((long)i);        }        else        {            tasks[i] = Task.Run(() => Fib(index));        }    }    // 等待所有任务完成    Task.WaitAll(tasks);    // 打印结果    for (var i = 0; i < n; i++)    {        Console.Write("{0} ", tasks[i].Result);    }    Console.ReadKey();}

    在这个示例中,我们使用Task数组来存储所有的任务。如果需要计算的是前两项,则直接使用Task.FromResult创建完成任务,否则使用Task.Run方法创建异步任务并调用Fib方法计算结果。在等待所有任务完成后,我们遍历Task数组,并使用Task.Result属性获取每个任务的结果并打印出来。

    运行的结果:

    0 1 1 2 3 5 8 13 21 34

    需要注意的是,在创建异步任务时,由于循环变量在闭包内的值是不确定的,因此需要将其赋值给另外一个变量,并在闭包内使用该变量。否则,所有任务可能会使用同一个循环变量的值,导致结果错误。

    示例3

    除了使用Task数组存储所有任务,还可以使用Task.Factory.StartNew方法创建并行任务。这个方法与Task.Run方法类似,都可以创建异步任务并提交给线程池。

    void Task3(){    long Factorial(int n)    {        if (n == 0) return 1;        return n * Factorial(n - 1);    }    const int n = 5; // 计算阶乘的数    var task = Task.Factory.StartNew(() => Factorial(n));    Console.WriteLine("计算阶乘...");    // 等待任务完成    task.Wait();    Console.WriteLine("{0}! = {1}", n, task.Result);    Console.ReadKey();}

    在这个示例中,我们使用Task.Factory.StartNew方法创建一个计算阶乘的异步任务,并等待任务完成后打印结果。

    运行结果:

    计算阶乘...
    5! = 120

    需要注意的是,尽管Task.Run和Task.Factory.StartNew方法都可以创建异步任务,但它们的行为略有不同。特别是,Task.Run方法总是使用TaskScheduler.Default作为任务调度器,而Task.Factory.StartNew方法可以指定任务调度器、任务类型和其他选项。因此,在选择使用哪种方法时,需要根据具体情况进行评估。

    示例4

    另一个使用Task的示例是异步读取文件。在这个示例中,我们使用Task.FromResult方法创建一个完成任务,并将文件内容作为结果返回。

    void Task4(){    const string filePath = "test.txt";    var task = Task.FromResult(File.ReadAllText(filePath)); // 只是方便举例,更好的代码应该是:File.ReadAllTextAsync(filePath);     Console.WriteLine("读取文件内容...");    // 等待任务完成    task.Wait();    Console.WriteLine("文件内容: {0}", task.Result);    Console.ReadKey();}

    在这个示例中,我们使用Task.FromResult方法创建一个完成任务,并通过File.ReadAllText方法读取文件内容并将其作为结果返回。在等待任务完成后,我们可以通过调用Task.Result属性来获取任务的结果。

    文中记事本请随意创建

    需要注意的是,在实际开发中,如果需要处理大型文件或需要执行长时间的I/O操作,则应该使用异步代码来避免阻塞UI线程。例如,在读取大型文件时,我们可以使用异步代码来避免阻塞UI线程,从而提高应用程序的性能和响应速度。

    示例5

    最后一个示例是使用Task和async/await实现异步任务。在这个示例中,我们将一个耗时的操作封装为异步方法,并使用async/await关键字来等待该操作完成。

    async Task Task5(){    async Task<string> LonGoperationAsync()    {        // 模拟耗时操作        await Task.Delay(TimeSpan.FromSeconds(3));        return "完成";    }    Console.WriteLine($"{DateTime.Now:yyyy-MM-dd HH:mm:ss.fff}开始耗时操作...");    // 等待异步方法完成    var result = await LongOperationAsync();    Console.WriteLine($"{DateTime.Now:yyyy-MM-dd HH:mm:ss.fff}耗时操作完成: {result}");    Console.ReadKey();}

    在这个示例中,我们使用async/await关键字将LongOperationAsync方法声明为异步方法,并使用await关键字等待Task.Delay操作完成。在主程序中,我们可以使用await关键字等待LongOperationAsync完成并获取其结果。

    2023-03-28 20:54:09.111开始耗时操作...
    2023-03-28 20:54:12.143耗时操作完成: 完成

    需要注意的是,在使用async/await关键字时,应该避免在异步方法内部使用阻塞线程的操作,否则可能会导致UI线程被阻塞。如果必须执行阻塞操作,可以将其放在不同的线程上执行,或者使用异步IO操作来避免阻塞线程。

    三、使用async/await关键字注意

    在使用async/await关键字时,还有一些细节需要注意,再给出两个示例。

    示例1

    示例代码如下所示:

    async Task Task6(){    async Task<string> LongOperationAsync(int id)    {        // 模拟耗时操作        await Task.Delay(TimeSpan.FromSeconds(1 + id));        return $"{DateTime.Now:ss.fff}完成 {id}";    }    Console.WriteLine($"{DateTime.Now:yyyy-MM-dd HH:mm:ss.fff}开始耗时操作...");    // 等待多个异步任务完成    var task1 = LongOperationAsync(1);    var task2 = LongOperationAsync(2);    var task3 = LongOperationAsync(3);    var results = await Task.WhenAll(task1, task2, task3);    var resultStr = string.Join(",", results);    Console.WriteLine($"{DateTime.Now:yyyy-MM-dd HH:mm:ss.fff}耗时操作完成: {resultStr}");    Console.ReadKey();}

    在这个示例中,我们使用Task.WhenAll方法等待多个异步任务完成,并使用Join方法将所有任务的结果连接起来作为最终结果。

    2023-03-28 21:15:42.855开始耗时操作...
    2023-03-28 21:15:46.894耗时操作完成: 44.888完成 1,45.883完成 2,46.893完成 3

    示例2

    另一个需要注意的问题是,在使用async/await关键字时,应该尽可能避免使用ConfigureAwait(false)方法。这个方法可以让异步操作不必恢复到原始的SynchronizationContext上,从而减少线程切换的开销和提高性能。

    然而,在某些情况下,如果在异步操作完成后需要返回到原始的SynchronizationContext上,使用ConfigureAwait(false)会导致调用者无法正确处理结果。因此,建议仅在确定不需要返回到原始的SynchronizationContext上时才使用ConfigureAwait(false)方法。

    示例代码: 假设我们有一个控制台应用程序,其中有两个异步方法:MethodAAsync()和MethodBAsync()。MethodAAsync()会等待1秒钟,然后返回一个字符串。MethodBAsync()会等待2秒钟,然后返回一个字符串。代码如下所示:

    async Task<string> MethodAAsync(){    await Task.Delay(1000);    return $"{DateTime.Now:ss.fff}>Hello";}async Task<string> MethodBAsync(){    await Task.Delay(2000);    return $"{DateTime.Now:ss.fff}>World";}

    现在,我们想要同时调用这两个方法,并将它们的结果合并成一个字符串。我们可以像下面这样编写代码:

    async Task<string> CombineResultsAAsync(){    var resultA = await MethodAAsync();    var resultB = await MethodBAsync();    return $"{DateTime.Now:yyyy-MM-dd HH:mm:ss.fff}: {resultA} | {resultB}";}

    这个代码看起来非常简单明了,但是它存在一个性能问题。当我们调用CombineResultsAAsync()方法时,第一个await操作将使执行上下文切换回原始SynchronizationContext(即主线程),因此我们的异步操作将在UI线程上运行。由于我们要等待1秒钟才能从MethodAAsync()中返回结果,因此UI线程将被阻塞,直到异步操作完成并且结果可用为止。

    这种情况下,我们可以使用ConfigureAwait(false)方法来指定不需要保留当前上下文的线程执行状态,从而让异步操作在一个线程池线程上运行。这可以通过下面的代码实现:

    async Task<string> CombineResultsBAsync(){    var resultA = await MethodAAsync().ConfigureAwait(false);    var resultB = await MethodBAsync().ConfigureAwait(false);    return $"{DateTime.Now:yyyy-MM-dd HH:mm:ss.fff}: {resultA} | {resultB}";}

    通过使用ConfigureAwait(false)方法,我们告诉异步操作不需要保留当前上下文的线程执行状态,这样异步操作就会在一个线程池线程上运行,而不是在UI线程上运行。这样做可以避免一些潜在的性能问题,因为我们的UI线程不会被阻塞,并且异步操作可以在一个新的线程池线程上运行。

    以上就是关于“C#怎么使用Task实现执行并行任务”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网精选频道。

    --结束END--

    本文标题: C#怎么使用Task实现执行并行任务

    本文链接: https://www.lsjlt.com/news/355427.html(转载时请注明来源链接)

    有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

    本篇文章演示代码以及资料文档资料下载

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档
    猜你喜欢
    • C#怎么使用Task实现执行并行任务
      这篇“C#怎么使用Task实现执行并行任务”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C#怎么使用Task实现执行并行任务...
      99+
      2023-07-05
    • C#怎么Task执行任务
      本篇内容介绍了“C#怎么Task执行任务”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!Task执行任务,等待任务完成代码://任务Func&...
      99+
      2023-07-02
    • C#使用Task实现执行并行任务的原理的示例详解
      目录一、Task执行并行任务的原理二、5个示例展示示例1示例2示例3示例4示例5三、使用async/await关键字注意示例1示例2四、总结一、Task执行并行任务的原理 使用Tas...
      99+
      2023-05-14
      C# Task执行并行任务 C# Task并行任务 C# Task
    • C#并行编程之Task任务
      目录一、任务与线程二、初识Task三、任务的结果.NET 4.5 :Task.Run四、连续任务五、分离嵌套任务六、子任务七、取消任务八、休眠:等待时间执行九、等待任务执行1、Tas...
      99+
      2024-04-02
    • C#使用Task实现并行编程
      故事背景 透着纱的窗外的阳光, 又是一个星期一. 慢慢来 一看时间, 还早, 那么蹦跶起来 穿衣刷牙洗脸 用代码来说的话, 应该是这样: // Program.cs using Sy...
      99+
      2024-04-02
    • C#如何Task执行任务,等待任务完成
      目录Task执行任务,等待任务完成C# Task任务队列需求基本的Task用法让Task任务按顺序执行使用异步委托解决UI界面卡死问题异步任务队列按顺序执行封装任务队列Task执行任...
      99+
      2024-04-02
    • 怎么在C#中利用Task执行异步
      这篇文章将为大家详细讲解有关怎么在C#中利用Task执行异步,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。为什么要使用 Task线程是创建并发的底层工具,因此具有一定的局限性。没有简单的方法...
      99+
      2023-06-14
    • C#任务并行Parellel.For和Parallel.ForEach怎么使用
      这篇文章主要介绍了C#任务并行Parellel.For和Parallel.ForEach怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C#任务并行Parellel.For和Parallel.ForEac...
      99+
      2023-07-02
    • C#如何使用Task执行异步操作
      目录为什么要使用 Task Task 和 Thread 区别 Task 介绍 Task 简单实现 Task 执行状态 1.等待(Wait) 2. 返回值 3. Task.Delay ...
      99+
      2024-04-02
    • golang同步执行任务怎么实现
      在Go中,可以通过goroutine和channels来实现任务的同步执行。具体步骤如下: 创建一个通道,用于任务之间的通信。 在...
      99+
      2023-10-22
      golang
    • php怎么实现定时执行任务
      在PHP中,可以使用以下几种方法来实现定时执行任务:1. 使用cron表达式:cron表达式是一种常用的定时任务调度语法,可以在操作...
      99+
      2023-08-12
      php
    • android定时执行任务怎么实现
      Android定时执行任务可以通过以下几种方式实现:1. 使用Handler和Timer:创建一个Handler对象,在Handle...
      99+
      2023-09-21
      android
    • Dart怎么实现多任务并行
      本篇内容介绍了“Dart怎么实现多任务并行”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!Isolate(隔离区域)Dart 是一种支持多任务...
      99+
      2023-07-05
    • 如何使用Systemd和Crontab在Linux系统中实现任务的并行执行
      要在Linux系统中实现任务的并行执行,可以结合使用Systemd和Crontab。下面是使用这两个工具的步骤:1. 创建并行执行的...
      99+
      2023-10-09
      Linux
    • 使用spring-task定时任务动态配置修改执行时间
      目录spring-task定时任务动态配置修改执行时间spring schedule 动态配置执行时间spring-task定时任务动态配置修改执行时间 因项目需要,几个定时任务需要...
      99+
      2024-04-02
    • Flask实现异步执行任务
      Flask 是 Python 中有名的轻量级同步 web 框架,在一些开发中,可能会遇到需要长时间处理的任务,此时就需要使用异步的方式来实现,让长时间任务在后台运行,先将本次请求的响...
      99+
      2024-04-02
    • 怎么使用Python3+pycuda实现执行简单GPU计算任务
      今天小编给大家分享一下怎么使用Python3+pycuda实现执行简单GPU计算任务的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一...
      99+
      2023-07-05
    • Laravel每秒执行定时任务怎么实现
      这篇文章主要介绍“Laravel每秒执行定时任务怎么实现”,在日常操作中,相信很多人在Laravel每秒执行定时任务怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Laravel每秒执行定时任务怎么实现...
      99+
      2023-07-04
    • 如何使用Python中的多线程进行任务并发执行
      如何使用Python中的多线程进行任务并发执行多线程是一种常用的并发编程技术,可以提高程序的执行效率。在Python中,使用多线程可以实现任务的并发执行,从而加快程序的运行速度。本文将介绍如何使用Python中的多线程进行任务的并发执行,并...
      99+
      2023-10-22
      并发执行 Python多线程
    • Java实现多任务执行助手
      本文实例为大家分享了Java实现多任务执行助手的具体代码,供大家参考,具体内容如下 1.多线程执行任务类 package com.visy.threadpool; import c...
      99+
      2022-11-13
      Java多任务执行助手 Java多任务执行 Java多任务
    软考高级职称资格查询
    编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
    • 官方手机版

    • 微信公众号

    • 商务合作