Task - 基于线程池的任务(在 System.Threading.Tasks 命名空间下)

  • 多 Task 的并行执行
  • Parallel - 并行计算(在 System.Threading.Tasks 命名空间下)

示例
1、演示 Task(基于线程池的任务)的基本应用
Thread/Tasks/TaskDemo.xaml

  1. <Page
  2. x:Class="XamlDemo.Thread.Tasks.TaskDemo"
  3. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  4. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  5. xmlns:local="using:XamlDemo.Thread.Tasks"
  6. xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  7. xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  8. mc:Ignorable="d">
  9.  
  10. <Grid Background="Transparent">
  11. <StackPanel Margin="120 0 0 0">
  12.  
  13. <TextBlock Name="lblMsg" FontSize="14.667" />
  14.  
  15. <Button Name="btnCreateTask" Content="执行一个没有返回值的任务" Click="btnCreateTask_Click_1" Margin="0 10 0 0" />
  16. <Button Name="btnCancelTask" Content="取消“执行一个没有返回值的任务”" Click="btnCancelTask_Click_1" Margin="0 10 0 0" />
  17.  
  18. <Button Name="btnCreateTaskWithReturn" Content="执行一个带返回值的任务" Click="btnCreateTaskWithReturn_Click_1" Margin="0 30 0 0" />
  19. <Button Name="btnCancelTaskWithReturn" Content="取消“执行一个带返回值的任务”" Click="btnCancelTaskWithReturn_Click_1" Margin="0 10 0 0" />
  20.  
  21. </StackPanel>
  22. </Grid>
  23. </Page>

Thread/Tasks/TaskDemo.xaml.cs

  1. /*
  2. * Task - 基于线程池的任务(在 System.Threading.Tasks 命名空间下)
  3. */
  4.  
  5. using System;
  6. using Windows.UI.Xaml;
  7. using Windows.UI.Xaml.Controls;
  8. using System.Threading.Tasks;
  9. using System.Threading;
  10. using Windows.UI.Core;
  11.  
  12. namespace XamlDemo.Thread.Tasks
  13. {
  14. public sealed partial class TaskDemo : Page
  15. {
  16. /*
  17. * CancellationTokenSource - 用于取消 CancellationToken
  18. * Token - 一个 CancellationToken 类型的对象,用于关联 Task
  19. * IsCancellationRequested - 是否收到了取消操作的请求
  20. * Cancel() - 发出取消操作的请求
  21. *
  22. * CancellationToken - 用于关联 Task,以便取消 Task
  23. * IsCancellationRequested - 是否收到了取消操作的请求
  24. * WaitHandle - 信号,可以通过 WaitHandle.WaitOne() 在当前线程等待
  25. * ThrowIfCancellationRequested() - 如果收到了取消操作的请求,则抛出一个 OperationCanceledException 异常
  26. */
  27. private CancellationTokenSource _cts;
  28.  
  29. public TaskDemo()
  30. {
  31. this.InitializeComponent();
  32. }
  33.  
  34. private void btnCreateTask_Click_1(object sender, RoutedEventArgs e)
  35. {
  36. _cts = new CancellationTokenSource();
  37.  
  38. // 实例化一个 Task,可随时通过 task.Status 获取任务状态
  39. Task task = new Task(
  40. (ctx) => // 任务所调用的方法,没有返回值
  41. {
  42. // 在当前线程上阻塞 3000 毫秒(当收到取消请求时会发出信号,停止阻塞)
  43. _cts.Token.WaitHandle.WaitOne(3000);
  44.  
  45. // 收到取消操作的请求后抛出一个 OperationCanceledException 异常,其会导致 task.IsCanceled 的值变为 true
  46. // 此处的代码等同于 _cts.Token.ThrowIfCancellationRequested();
  47. if (_cts.IsCancellationRequested)
  48. throw new OperationCanceledException(_cts.Token);
  49. },
  50. null, // 上下文对象,task.AsyncState 可获取到此对象,上面的 ctx 也可获取到此对象
  51. _cts.Token // 关联的 CancellationToken 对象,用于取消操作
  52. );
  53.  
  54. // 开始执行任务
  55. task.Start();
  56. // task.Wait(); 在当前线程上等待任务执行完
  57. lblMsg.Text = "执行了一个没有返回值的任务,3 秒后执行完毕";
  58.  
  59. // 任务执行完毕后的处理(注:ContinueWith 方法支持任意次回调,即可以写多个 task.ContinueWith() 都会被回调)
  60. task.ContinueWith(
  61. (ctx) => // 任务执行完毕后所调用的方法
  62. {
  63. if (ctx.IsCanceled) // 任务被取消
  64. {
  65. var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
  66. () =>
  67. {
  68. lblMsg.Text += Environment.NewLine;
  69. lblMsg.Text += "取消了“执行一个没有返回值的任务”";
  70. });
  71. }
  72. if (ctx.IsFaulted) // 任务发生了一个未处理异常
  73. {
  74. var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
  75. () =>
  76. {
  77. lblMsg.Text += Environment.NewLine;
  78. lblMsg.Text += "“执行一个没有返回值的任务”发生了一个未处理异常";
  79. });
  80. }
  81. if (ctx.IsCompleted) // 任务已完成(任务成功地执行完毕或被取消或发生了未处理异常都会 ctx.IsCompleted == true)
  82. {
  83. var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
  84. () =>
  85. {
  86. lblMsg.Text += Environment.NewLine;
  87. lblMsg.Text += "“执行一个没有返回值的任务”执行完成,taskId: " + ctx.Id.ToString();
  88. });
  89. }
  90. });
  91. }
  92.  
  93. private void btnCancelTask_Click_1(object sender, RoutedEventArgs e)
  94. {
  95. // 发出取消操作的请求
  96. _cts.Cancel();
  97. // _cts.CancelAfter(1000); // 1000 毫秒后发出取消操作的请求
  98. }
  99.  
  100. private void btnCreateTaskWithReturn_Click_1(object sender, RoutedEventArgs e)
  101. {
  102. _cts = new CancellationTokenSource();
  103.  
  104. Func<object, string> handler = delegate(object state) // state 是传递过来的上下文对象
  105. {
  106. // 在当前线程上阻塞 3000 毫秒(当收到取消请求时会发出信号,停止阻塞)
  107. _cts.Token.WaitHandle.WaitOne(3000);
  108.  
  109. // 收到取消操作的请求后抛出一个 OperationCanceledException 异常,其会导致 task.IsCanceled 的值变为 true
  110. // 此处的代码等同于 _cts.Token.ThrowIfCancellationRequested();
  111. if (_cts.IsCancellationRequested)
  112. throw new OperationCanceledException(_cts.Token);
  113.  
  114. return "我是“执行一个带返回值的任务”的返回值";
  115. };
  116.  
  117. // Task.Factory.StartNew() - 创建任务并马上执行,可随时通过 task.Status 获取任务状态
  118. // Task.Run() 同样是创建任务并马上执行
  119. Task<string> task = Task.Factory.StartNew<string>(
  120. handler, // 任务所调用的方法,带返回值
  121. null, // 上下文对象,task.AsyncState 可获取到此对象
  122. _cts.Token // 关联的 CancellationToken 对象,用于取消操作
  123. );
  124. lblMsg.Text = "执行了一个带返回值的任务,3 秒后执行完毕";
  125.  
  126. // 任务执行完毕后的处理(注:ContinueWith 方法支持任意次回调,即可以写多个 task.ContinueWith() 都会被回调)
  127. task.ContinueWith(
  128. (ctx) =>
  129. {
  130. if (ctx.IsCanceled) // 任务被取消
  131. {
  132. var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
  133. () =>
  134. {
  135. lblMsg.Text += Environment.NewLine;
  136. lblMsg.Text += "取消了“执行一个带返回值的任务”";
  137. });
  138. }
  139. if (ctx.IsFaulted) // 任务发生了一个未处理异常
  140. {
  141. var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
  142. () =>
  143. {
  144. lblMsg.Text += Environment.NewLine;
  145. lblMsg.Text += "“执行一个带返回值的任务”发生了一个未处理异常";
  146. });
  147. }
  148. if (ctx.IsCompleted) // 任务已完成(任务成功地执行完毕或被取消或发生了未处理异常都会 ctx.IsCompleted == true)
  149. {
  150. var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
  151. () =>
  152. {
  153. lblMsg.Text += Environment.NewLine;
  154. lblMsg.Text += "“执行一个带返回值的任务”执行完成,taskId: " + ctx.Id.ToString();
  155. });
  156.  
  157. // 当任务成功地执行完毕时,输出任务的返回值
  158. if (!ctx.IsCanceled && !ctx.IsFaulted)
  159. {
  160. ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
  161. () =>
  162. {
  163. lblMsg.Text += Environment.NewLine;
  164. // 任务的返回值
  165. lblMsg.Text += ctx.Result;
  166. });
  167. }
  168. }
  169. });
  170. }
  171.  
  172. private void btnCancelTaskWithReturn_Click_1(object sender, RoutedEventArgs e)
  173. {
  174. // 发出取消操作的请求
  175. _cts.Cancel();
  176. // _cts.CancelAfter(1000); // 1000 毫秒后发出取消操作的请求
  177. }
  178. }
  179. }

2、演示多 Task 的并行执行
Thread/Tasks/MultiTask.xaml

  1. <Page
  2. x:Class="XamlDemo.Thread.Tasks.MultiTask"
  3. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  4. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  5. xmlns:local="using:XamlDemo.Thread.Tasks"
  6. xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  7. xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  8. mc:Ignorable="d">
  9.  
  10. <Grid Background="Transparent">
  11. <StackPanel Margin="120 0 0 0">
  12.  
  13. <TextBlock Name="lblMsg" FontSize="14.667" />
  14.  
  15. <Button Name="btnCreateMultiTask" Content="任务并行执行" Click="btnCreateMultiTask_Click_1" Margin="0 10 0 0" />
  16.  
  17. </StackPanel>
  18. </Grid>
  19. </Page>

Thread/Tasks/MultiTask.xaml.cs

  1. /*
  2. * 演示多 Task 的并行执行
  3. *
  4. * 注:
  5. * 本例中同时创建了三个任务 task1, task2, task3,但是由于 Task 是基于线程池的,所以三个任务的启动时间是不一样的,启动顺序是不一定的
  6. * 启动顺序可能是 task1->task2->task3,也可能是 task3->task2->task1,也可能是 task2->task3->task1,等等等等都有可能,是不一定的
  7. */
  8.  
  9. using System;
  10. using System.Threading.Tasks;
  11. using Windows.UI.Xaml;
  12. using Windows.UI.Xaml.Controls;
  13.  
  14. namespace XamlDemo.Thread.Tasks
  15. {
  16. public sealed partial class MultiTask : Page
  17. {
  18. private static int _count = 0;
  19.  
  20. public MultiTask()
  21. {
  22. this.InitializeComponent();
  23. }
  24.  
  25. private void btnCreateMultiTask_Click_1(object sender, RoutedEventArgs e)
  26. {
  27. // 创建并执行任务1
  28. Task task1 = Task.Run(
  29. () =>
  30. {
  31. new System.Threading.ManualResetEvent(false).WaitOne(3000);
  32. System.Threading.Interlocked.Increment(ref _count);
  33. });
  34. // 创建并执行任务2
  35. Task task2 = Task.Run(
  36. () =>
  37. {
  38. new System.Threading.ManualResetEvent(false).WaitOne(3000);
  39. System.Threading.Interlocked.Increment(ref _count);
  40. });
  41. // 创建并执行任务3
  42. Task task3 = Task.Run(
  43. () =>
  44. {
  45. new System.Threading.ManualResetEvent(false).WaitOne(3000);
  46. System.Threading.Interlocked.Increment(ref _count);
  47. });
  48.  
  49. // 将所有任务合成一个 Task 对象,不会阻塞 UI 线程,通过 task.ContinueWith() 获取结果
  50. Task task = Task.WhenAll(task1, task2, task3);
  51. // Task.WaitAll(task1, task2, task3); 等待所有任务完成,会阻塞 UI 线程
  52.  
  53. DateTime dt = DateTime.Now;
  54.  
  55. // task 执行完毕后的处理,即所有任务执行完毕后的处理
  56. task.ContinueWith(
  57. (ctx) =>
  58. {
  59. var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
  60. () =>
  61. {
  62. lblMsg.Text = "count: " + _count.ToString() + ", 执行时间: " + (DateTime.Now - dt).TotalSeconds.ToString() + "秒";
  63. });
  64. });
  65. }
  66. }
  67. }

3、演示 Parallel(并行计算)的基本应用
Thread/Tasks/ParallelDemo.xaml

  1. <Page
  2. x:Class="XamlDemo.Thread.Tasks.ParallelDemo"
  3. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  4. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  5. xmlns:local="using:XamlDemo.Thread.Tasks"
  6. xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  7. xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  8. mc:Ignorable="d">
  9.  
  10. <Grid Background="Transparent">
  11. <StackPanel Margin="120 0 0 0">
  12.  
  13. <TextBlock Name="lblMsg1" FontSize="14.667" />
  14. <TextBlock Name="lblMsg2" FontSize="14.667" />
  15.  
  16. <Button Name="btnCreateParallel" Content="执行一个并行运算(Parallel)" Click="btnCreateParallel_Click_1" Margin="0 10 0 0" />
  17.  
  18. <Button Name="btnCancelParallel" Content="取消" Click="btnCancelParallel_Click_1" Margin="0 10 0 0" />
  19.  
  20. </StackPanel>
  21. </Grid>
  22. </Page>

Thread/Tasks/ParallelDemo.xaml.cs

  1. /*
  2. * Parallel - 并行计算(在 System.Threading.Tasks 命名空间下)
  3. *
  4. * Parallel.For() - for 循环的并行运算
  5. * Parallel.ForEach() - foreach 循环的并行运算
  6. * Parallel.Invoke() - 并行调用多个 Action
  7. * PLINQ - LINQ to Object 的并行运算
  8. *
  9. *
  10. * 本例通过 Parallel.Invoke() 来演示并行运算
  11. * 其它并行运算的说明参见:http://www.cnblogs.com/webabcd/archive/2010/06/03/1750449.html
  12. */
  13.  
  14. using System.Threading;
  15. using System.Threading.Tasks;
  16. using Windows.UI.Xaml;
  17. using Windows.UI.Xaml.Controls;
  18.  
  19. namespace XamlDemo.Thread.Tasks
  20. {
  21. public sealed partial class ParallelDemo : Page
  22. {
  23. private CancellationTokenSource _cts;
  24.  
  25. public ParallelDemo()
  26. {
  27. this.InitializeComponent();
  28. }
  29.  
  30. private void btnCreateParallel_Click_1(object sender, RoutedEventArgs e)
  31. {
  32. if (_cts != null)
  33. _cts.Cancel();
  34. _cts = new CancellationTokenSource();
  35.  
  36. // Parallel 的相关配置
  37. ParallelOptions parallelOptions = new ParallelOptions()
  38. {
  39. CancellationToken = _cts.Token, // Parallel 关联的 CancellationToken 对象,用于取消操作
  40. MaxDegreeOfParallelism = 10 // Parallel 的最大并行数
  41. };
  42.  
  43. // 并行执行多个 Action(不支持 Func)
  44. Parallel.Invoke(
  45. parallelOptions,
  46. () =>
  47. Task1(parallelOptions.CancellationToken),
  48. () =>
  49. Task2(parallelOptions.CancellationToken));
  50. }
  51.  
  52. // Action 1
  53. private void Task1(CancellationToken token)
  54. {
  55. Task task = Task.Factory.StartNew(
  56. () =>
  57. {
  58. int count = 0;
  59. // 未被取消则一直运行
  60. while (!token.IsCancellationRequested)
  61. {
  62. count++;
  63.  
  64. var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
  65. () =>
  66. {
  67. lblMsg1.Text = "count1: " + count.ToString();
  68. });
  69.  
  70. token.WaitHandle.WaitOne(100);
  71. }
  72. },
  73. token);
  74. }
  75.  
  76. // Action 2
  77. private void Task2(CancellationToken token)
  78. {
  79. Task task = Task.Factory.StartNew(
  80. () =>
  81. {
  82. int count = 0;
  83. // 未被取消则一直运行
  84. while (!token.IsCancellationRequested)
  85. {
  86. count++;
  87.  
  88. var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
  89. () =>
  90. {
  91. lblMsg2.Text = "count2: " + count.ToString();
  92. });
  93.  
  94. token.WaitHandle.WaitOne(100);
  95. }
  96. },
  97. token);
  98. }
  99.  
  100. // 取消并行运算
  101. private void btnCancelParallel_Click_1(object sender, RoutedEventArgs e)
  102. {
  103. if (_cts != null)
  104. _cts.Cancel();
  105. }
  106. }
  107. }

多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)的更多相关文章

  1. 重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)

    [源码下载] 重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel) 作者:webabcd 介绍重新想象 W ...

  2. SQL Server调优系列基础篇(并行运算总结篇二)

    前言 上一篇文章我们介绍了查看查询计划的并行运行方式. 本篇我们接着分析SQL Server的并行运算. 闲言少叙,直接进入本篇的正题. 技术准备 同前几篇一样,基于SQL Server2008R2版 ...

  3. c#中@标志的作用 C#通过序列化实现深表复制 细说并发编程-TPL 大数据量下DataTable To List效率对比 【转载】C#工具类:实现文件操作File的工具类 异步多线程 Async .net 多线程 Thread ThreadPool Task .Net 反射学习

    c#中@标志的作用   参考微软官方文档-特殊字符@,地址 https://docs.microsoft.com/zh-cn/dotnet/csharp/language-reference/toke ...

  4. 转载 .Net多线程编程—任务Task https://www.cnblogs.com/hdwgxz/p/6258014.html

    .Net多线程编程—任务Task   1 System.Threading.Tasks.Task简介 一个Task表示一个异步操作,Task的创建和执行是独立的. 只读属性: 返回值 名称 说明 ob ...

  5. 异步多线程 Thread ThreadPool Task

    一.线程 Thread ThreadPool 线程是Windows任务调度的最小单位,线程是程序中的一个执行流,每个线程都有自己的专有寄存器(栈指针.程序计数器等),但代码区是共享的,即不同的线程可以 ...

  6. C# 多线程六之Task(任务)三之任务工厂

    1.知识回顾,简要概述 前面两篇关于Task的随笔,C# 多线程五之Task(任务)一 和 C# 多线程六之Task(任务)二,介绍了关于Task的一些基本的用法,以及一些使用的要点,如果都看懂了,本 ...

  7. C#多线程(14):任务基础②

    目录 判断任务状态 再说父子任务 组合任务/延续任务 复杂的延续任务 并行(异步)处理任务 并行(同步)处理任务 并行任务的 Task.WhenAny 并行任务状态 循环中值变化问题 定时任务 Tas ...

  8. C#多线程实现方法——Task/Task.Factary

    原文:C#多线程实现方法--Task/Task.Factary Task 使用 Task以及Task.Factory都是在.Net 4引用的.Task跟Thread很类似,通过下面例子可以看到. st ...

  9. .NET 并行(多核)编程系列之六 Task基础部分完结篇

    原文:.NET 并行(多核)编程系列之六 Task基础部分完结篇 .NET 并行(多核)编程系列之六 Task基础部分完结篇 前言:之前的文章介绍了了并行编程的一些基本的,也注重的讲述了Task的一些 ...

随机推荐

  1. VMware安装CentOS时,无法以图形界面安装解决办法

    有的同学问: 用虚拟机软件(vmware.VirtualBox)安装CentOS系统时, 安装过程中没有中文,也没有出现图形界面,都是以命令行方式去安装, 有时候又会出现图形界面,不知道哪里配置的问题 ...

  2. 关于IOS中safari下的select下拉菜单,文字过长不换行的问题

    今天遇到下图这种问题,文字过长,显示不全.折腾了老半天,在网上搜了半天也找不到解决方案. 于是问了下同事,同事提到了<optgroup>,这个标签厉害. <optgroup> ...

  3. AlloyTouch 0.2.0发布--鱼和熊掌兼得

    原文链接:https://github.com/AlloyTeam/AlloyTouch/wiki/AlloyTouch-0.2.0 背景 公司师姐昨日在KM发了篇长文,主要结论RAF+transfo ...

  4. Android EventBus 3.0.0 使用总结

    转载请标明出处:http://www.cnblogs.com/zhaoyanjun/p/6039221.html 本文出自[赵彦军的博客] 前言 EventBus框架 EventBus是一个通用的叫法 ...

  5. ios native工程集成react-native的demo

    react-native看到了给现有工程添加react-native环境的时候碰到一个问题: 如何往工程中添加 package.json文件,以及node_modules是怎么来的? 我开始的时候以为 ...

  6. Atitit.web ui  组件化 vs  mvc

    Atitit.web ui  组件化 vs  mvc   组件化  与 mvc并不矛盾..单双方适用的地方有所不同.. React推荐以组件的方式去重新思考UI构成,将UI上每一个功能相对独立的模块定 ...

  7. Quartz2.0以上版本的单机和集群

    (一)Quartz单机 1.Quartz简介 Quartz是一个完全由java编写的开源作业调度框架,能实现业务的定时调度.Quartz主要有三个核心调度器.任务和触发器: ①任务-JobDetail ...

  8. pscp工具上传或取回文件、文件夹

    1.把服务器上的/root/dir目录取回本地"C:\My Documents\data\"目录 C:\>pscp.exe -r root@IP:/root/dir &quo ...

  9. Scala基础语法 (一)

    如果你之前是一名 Java 程序员,并了解 Java 语言的基础知识,那么你能很快学会 Scala 的基础语法. Scala 与 Java 的最大区别是:Scala 语句末尾的分号 ; 是可选的. 我 ...

  10. Ubuntu 14.04 LTS下安装Google Chrome浏览器

    在Ubuntu 14.04下安装Google Chrome浏览器非常简单,只要到Chrome的网站下载Deb安装包并进行安装即可.当然你也可以使用APT软件包管理器来安装Google Chrome浏览 ...