重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)
作者:webabcd
介绍
重新想象 Windows 8 Store Apps 之 任务
- Task - 基于线程池的任务(在 System.Threading.Tasks 命名空间下)
- 多 Task 的并行执行
- Parallel - 并行计算(在 System.Threading.Tasks 命名空间下)
示例
1、演示 Task(基于线程池的任务)的基本应用
Thread/Tasks/TaskDemo.xaml
<Page
x:Class="XamlDemo.Thread.Tasks.TaskDemo"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:XamlDemo.Thread.Tasks"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"> <Grid Background="Transparent">
<StackPanel Margin="120 0 0 0"> <TextBlock Name="lblMsg" FontSize="14.667" /> <Button Name="btnCreateTask" Content="执行一个没有返回值的任务" Click="btnCreateTask_Click_1" Margin="0 10 0 0" />
<Button Name="btnCancelTask" Content="取消“执行一个没有返回值的任务”" Click="btnCancelTask_Click_1" Margin="0 10 0 0" /> <Button Name="btnCreateTaskWithReturn" Content="执行一个带返回值的任务" Click="btnCreateTaskWithReturn_Click_1" Margin="0 30 0 0" />
<Button Name="btnCancelTaskWithReturn" Content="取消“执行一个带返回值的任务”" Click="btnCancelTaskWithReturn_Click_1" Margin="0 10 0 0" /> </StackPanel>
</Grid>
</Page>
Thread/Tasks/TaskDemo.xaml.cs
/*
* Task - 基于线程池的任务(在 System.Threading.Tasks 命名空间下)
*/ using System;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls;
using System.Threading.Tasks;
using System.Threading;
using Windows.UI.Core; namespace XamlDemo.Thread.Tasks
{
public sealed partial class TaskDemo : Page
{
/*
* CancellationTokenSource - 用于取消 CancellationToken
* Token - 一个 CancellationToken 类型的对象,用于关联 Task
* IsCancellationRequested - 是否收到了取消操作的请求
* Cancel() - 发出取消操作的请求
*
* CancellationToken - 用于关联 Task,以便取消 Task
* IsCancellationRequested - 是否收到了取消操作的请求
* WaitHandle - 信号,可以通过 WaitHandle.WaitOne() 在当前线程等待
* ThrowIfCancellationRequested() - 如果收到了取消操作的请求,则抛出一个 OperationCanceledException 异常
*/
private CancellationTokenSource _cts; public TaskDemo()
{
this.InitializeComponent();
} private void btnCreateTask_Click_1(object sender, RoutedEventArgs e)
{
_cts = new CancellationTokenSource(); // 实例化一个 Task,可随时通过 task.Status 获取任务状态
Task task = new Task(
(ctx) => // 任务所调用的方法,没有返回值
{
// 在当前线程上阻塞 3000 毫秒(当收到取消请求时会发出信号,停止阻塞)
_cts.Token.WaitHandle.WaitOne(); // 收到取消操作的请求后抛出一个 OperationCanceledException 异常,其会导致 task.IsCanceled 的值变为 true
// 此处的代码等同于 _cts.Token.ThrowIfCancellationRequested();
if (_cts.IsCancellationRequested)
throw new OperationCanceledException(_cts.Token);
},
null, // 上下文对象,task.AsyncState 可获取到此对象,上面的 ctx 也可获取到此对象
_cts.Token // 关联的 CancellationToken 对象,用于取消操作
); // 开始执行任务
task.Start();
// task.Wait(); 在当前线程上等待任务执行完
lblMsg.Text = "执行了一个没有返回值的任务,3 秒后执行完毕"; // 任务执行完毕后的处理(注:ContinueWith 方法支持任意次回调,即可以写多个 task.ContinueWith() 都会被回调)
task.ContinueWith(
(ctx) => // 任务执行完毕后所调用的方法
{
if (ctx.IsCanceled) // 任务被取消
{
var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
() =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += "取消了“执行一个没有返回值的任务”";
});
}
if (ctx.IsFaulted) // 任务发生了一个未处理异常
{
var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
() =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += "“执行一个没有返回值的任务”发生了一个未处理异常";
});
}
if (ctx.IsCompleted) // 任务已完成(任务成功地执行完毕或被取消或发生了未处理异常都会 ctx.IsCompleted == true)
{
var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
() =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += "“执行一个没有返回值的任务”执行完成,taskId: " + ctx.Id.ToString();
});
}
});
} private void btnCancelTask_Click_1(object sender, RoutedEventArgs e)
{
// 发出取消操作的请求
_cts.Cancel();
// _cts.CancelAfter(1000); // 1000 毫秒后发出取消操作的请求
} private void btnCreateTaskWithReturn_Click_1(object sender, RoutedEventArgs e)
{
_cts = new CancellationTokenSource(); Func<object, string> handler = delegate(object state) // state 是传递过来的上下文对象
{
// 在当前线程上阻塞 3000 毫秒(当收到取消请求时会发出信号,停止阻塞)
_cts.Token.WaitHandle.WaitOne(); // 收到取消操作的请求后抛出一个 OperationCanceledException 异常,其会导致 task.IsCanceled 的值变为 true
// 此处的代码等同于 _cts.Token.ThrowIfCancellationRequested();
if (_cts.IsCancellationRequested)
throw new OperationCanceledException(_cts.Token); return "我是“执行一个带返回值的任务”的返回值";
}; // Task.Factory.StartNew() - 创建任务并马上执行,可随时通过 task.Status 获取任务状态
// Task.Run() 同样是创建任务并马上执行
Task<string> task = Task.Factory.StartNew<string>(
handler, // 任务所调用的方法,带返回值
null, // 上下文对象,task.AsyncState 可获取到此对象
_cts.Token // 关联的 CancellationToken 对象,用于取消操作
);
lblMsg.Text = "执行了一个带返回值的任务,3 秒后执行完毕"; // 任务执行完毕后的处理(注:ContinueWith 方法支持任意次回调,即可以写多个 task.ContinueWith() 都会被回调)
task.ContinueWith(
(ctx) =>
{
if (ctx.IsCanceled) // 任务被取消
{
var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
() =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += "取消了“执行一个带返回值的任务”";
});
}
if (ctx.IsFaulted) // 任务发生了一个未处理异常
{
var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
() =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += "“执行一个带返回值的任务”发生了一个未处理异常";
});
}
if (ctx.IsCompleted) // 任务已完成(任务成功地执行完毕或被取消或发生了未处理异常都会 ctx.IsCompleted == true)
{
var ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
() =>
{
lblMsg.Text += Environment.NewLine;
lblMsg.Text += "“执行一个带返回值的任务”执行完成,taskId: " + ctx.Id.ToString();
}); // 当任务成功地执行完毕时,输出任务的返回值
if (!ctx.IsCanceled && !ctx.IsFaulted)
{
ignored = Dispatcher.RunAsync(CoreDispatcherPriority.High,
() =>
{
lblMsg.Text += Environment.NewLine;
// 任务的返回值
lblMsg.Text += ctx.Result;
});
}
}
});
} private void btnCancelTaskWithReturn_Click_1(object sender, RoutedEventArgs e)
{
// 发出取消操作的请求
_cts.Cancel();
// _cts.CancelAfter(1000); // 1000 毫秒后发出取消操作的请求
}
}
}
2、演示多 Task 的并行执行
Thread/Tasks/MultiTask.xaml
<Page
x:Class="XamlDemo.Thread.Tasks.MultiTask"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:XamlDemo.Thread.Tasks"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"> <Grid Background="Transparent">
<StackPanel Margin="120 0 0 0"> <TextBlock Name="lblMsg" FontSize="14.667" /> <Button Name="btnCreateMultiTask" Content="任务并行执行" Click="btnCreateMultiTask_Click_1" Margin="0 10 0 0" /> </StackPanel>
</Grid>
</Page>
Thread/Tasks/MultiTask.xaml.cs
/*
* 演示多 Task 的并行执行
*
* 注:
* 本例中同时创建了三个任务 task1, task2, task3,但是由于 Task 是基于线程池的,所以三个任务的启动时间是不一样的,启动顺序是不一定的
* 启动顺序可能是 task1->task2->task3,也可能是 task3->task2->task1,也可能是 task2->task3->task1,等等等等都有可能,是不一定的
*/ using System;
using System.Threading.Tasks;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls; namespace XamlDemo.Thread.Tasks
{
public sealed partial class MultiTask : Page
{
private static int _count = ; public MultiTask()
{
this.InitializeComponent();
} private void btnCreateMultiTask_Click_1(object sender, RoutedEventArgs e)
{
// 创建并执行任务1
Task task1 = Task.Run(
() =>
{
new System.Threading.ManualResetEvent(false).WaitOne();
System.Threading.Interlocked.Increment(ref _count);
});
// 创建并执行任务2
Task task2 = Task.Run(
() =>
{
new System.Threading.ManualResetEvent(false).WaitOne();
System.Threading.Interlocked.Increment(ref _count);
});
// 创建并执行任务3
Task task3 = Task.Run(
() =>
{
new System.Threading.ManualResetEvent(false).WaitOne();
System.Threading.Interlocked.Increment(ref _count);
}); // 将所有任务合成一个 Task 对象,不会阻塞 UI 线程,通过 task.ContinueWith() 获取结果
Task task = Task.WhenAll(task1, task2, task3);
// Task.WaitAll(task1, task2, task3); 等待所有任务完成,会阻塞 UI 线程 DateTime dt = DateTime.Now; // task 执行完毕后的处理,即所有任务执行完毕后的处理
task.ContinueWith(
(ctx) =>
{
var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
() =>
{
lblMsg.Text = "count: " + _count.ToString() + ", 执行时间: " + (DateTime.Now - dt).TotalSeconds.ToString() + "秒";
});
});
}
}
}
3、演示 Parallel(并行计算)的基本应用
Thread/Tasks/ParallelDemo.xaml
<Page
x:Class="XamlDemo.Thread.Tasks.ParallelDemo"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:XamlDemo.Thread.Tasks"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"> <Grid Background="Transparent">
<StackPanel Margin="120 0 0 0"> <TextBlock Name="lblMsg1" FontSize="14.667" />
<TextBlock Name="lblMsg2" FontSize="14.667" /> <Button Name="btnCreateParallel" Content="执行一个并行运算(Parallel)" Click="btnCreateParallel_Click_1" Margin="0 10 0 0" /> <Button Name="btnCancelParallel" Content="取消" Click="btnCancelParallel_Click_1" Margin="0 10 0 0" /> </StackPanel>
</Grid>
</Page>
Thread/Tasks/ParallelDemo.xaml.cs
/*
* Parallel - 并行计算(在 System.Threading.Tasks 命名空间下)
*
* Parallel.For() - for 循环的并行运算
* Parallel.ForEach() - foreach 循环的并行运算
* Parallel.Invoke() - 并行调用多个 Action
* PLINQ - LINQ to Object 的并行运算
*
*
* 本例通过 Parallel.Invoke() 来演示并行运算
* 其它并行运算的说明参见:http://www.cnblogs.com/webabcd/archive/2010/06/03/1750449.html
*/ using System.Threading;
using System.Threading.Tasks;
using Windows.UI.Xaml;
using Windows.UI.Xaml.Controls; namespace XamlDemo.Thread.Tasks
{
public sealed partial class ParallelDemo : Page
{
private CancellationTokenSource _cts; public ParallelDemo()
{
this.InitializeComponent();
} private void btnCreateParallel_Click_1(object sender, RoutedEventArgs e)
{
if (_cts != null)
_cts.Cancel();
_cts = new CancellationTokenSource(); // Parallel 的相关配置
ParallelOptions parallelOptions = new ParallelOptions()
{
CancellationToken = _cts.Token, // Parallel 关联的 CancellationToken 对象,用于取消操作
MaxDegreeOfParallelism = // Parallel 的最大并行数
}; // 并行执行多个 Action(不支持 Func)
Parallel.Invoke(
parallelOptions,
() =>
Task1(parallelOptions.CancellationToken),
() =>
Task2(parallelOptions.CancellationToken));
} // Action 1
private void Task1(CancellationToken token)
{
Task task = Task.Factory.StartNew(
() =>
{
int count = ;
// 未被取消则一直运行
while (!token.IsCancellationRequested)
{
count++; var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
() =>
{
lblMsg1.Text = "count1: " + count.ToString();
}); token.WaitHandle.WaitOne();
}
},
token);
} // Action 2
private void Task2(CancellationToken token)
{
Task task = Task.Factory.StartNew(
() =>
{
int count = ;
// 未被取消则一直运行
while (!token.IsCancellationRequested)
{
count++; var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
() =>
{
lblMsg2.Text = "count2: " + count.ToString();
}); token.WaitHandle.WaitOne();
}
},
token);
} // 取消并行运算
private void btnCancelParallel_Click_1(object sender, RoutedEventArgs e)
{
if (_cts != null)
_cts.Cancel();
}
}
}
OK
[源码下载]
重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)的更多相关文章
- 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)
Task - 基于线程池的任务(在 System.Threading.Tasks 命名空间下) 多 Task 的并行执行 Parallel - 并行计算(在 System.Threading.Task ...
- 重新想象 Windows 8 Store Apps (42) - 多线程之线程池: 延迟执行, 周期执行, 在线程池中找一个线程去执行指定的方法
[源码下载] 重新想象 Windows 8 Store Apps (42) - 多线程之线程池: 延迟执行, 周期执行, 在线程池中找一个线程去执行指定的方法 作者:webabcd 介绍重新想象 Wi ...
- 重新想象 Windows 8 Store Apps (44) - 多线程之异步编程: 经典和最新的异步编程模型, IAsyncInfo 与 Task 相互转换
[源码下载] 重新想象 Windows 8 Store Apps (44) - 多线程之异步编程: 经典和最新的异步编程模型, IAsyncInfo 与 Task 相互转换 作者:webabcd 介绍 ...
- 重新想象 Windows 8 Store Apps (45) - 多线程之异步编程: IAsyncAction, IAsyncOperation, IAsyncActionWithProgress, IAsyncOperationWithProgress
[源码下载] 重新想象 Windows 8 Store Apps (45) - 多线程之异步编程: IAsyncAction, IAsyncOperation, IAsyncActionWithPro ...
- 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLock
[源码下载] 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLoc ...
- 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEvent, AutoResetEvent
[源码下载] 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEve ...
- 重新想象 Windows 8 Store Apps (48) - 多线程之其他辅助类: SpinWait, SpinLock, Volatile, SynchronizationContext, CoreDispatcher, ThreadLocal, ThreadStaticAttribute
[源码下载] 重新想象 Windows 8 Store Apps (48) - 多线程之其他辅助类: SpinWait, SpinLock, Volatile, SynchronizationCont ...
- 重新想象 Windows 8 Store Apps (9) - 控件之 ScrollViewer 基础
原文:重新想象 Windows 8 Store Apps (9) - 控件之 ScrollViewer 基础 [源码下载] 重新想象 Windows 8 Store Apps (9) - 控件之 Sc ...
- 重新想象 Windows 8 Store Apps 系列文章索引
[源码下载][重新想象 Windows 8.1 Store Apps 系列文章] 重新想象 Windows 8 Store Apps 系列文章索引 作者:webabcd 1.重新想象 Windows ...
随机推荐
- MSSql使用SQL语句快速查看表对的就说明,及表字段描述及字段类型
--表描述 SELECT tbs.name 表名,ds.value 描述 FROM sys.extended_properties ds LEFT JOIN sysobjects tbs ON ds. ...
- Mac下MySQL卸载方法 转载
mac下mysql的DMG格式安装内有安装文件,却没有卸载文件……很郁闷的事. 网上搜了一下,发现给的方法原来得手动去删. 很多文章记述要删的文件不完整,后来在stackoverflow这里发现了一个 ...
- MD5 加密的两种方法
System.Security.Cryptography.MD5CryptoServiceProvider md5 = new System.Security.Cryptography.MD5Cryp ...
- Schema Workbench 开发mdx和模式文件
一.前言 安装了saiku之后,每次修改schema文件,非常耗时,每次都要经历若干步骤:修改xml.上传.重启才能生效,并且非常不利于学习和理解MDX和模式文件,踌躇之际,发现了这个工具,十分小巧方 ...
- [PaPaPa][需求说明书][V2.0]
前 言 大家好,我是“今晚打老虎”. 什么? 你问我为什么这次亮字号了? 还不是因为哥太出名了,即使我不亮你们也知道是我写的了. 自从发布了V1.0版本之后.群里又进来好多人.30K大大分发的任务 ...
- Android之输入框光标和Hint的位置
如图所示,要实现这一的需求,一般人的布局方式就是左边一button,右边一button,中间一个EditText,为了输入框的响应触摸范围更大往往不会把宽度设置为wrap_content,要么设置成m ...
- 用qt代码怎样编写图片保存格式[qt4.6]
用qt代码怎样编写图片保存格式 qt提供了多个保存图片的接口,比较常用的接口如下 bool QPixmap::save ( const QString & fileName, const ch ...
- 【译文】 C#面向对象的基本概念 (Basic C# OOP Concept) 第一部分(类,对象,变量,方法,访问修饰符)
译文出处:http://www.codeproject.com/Articles/838365/Basic-Csharp-OOP-Concept 相关文档:http://files.cnblogs.c ...
- oracle 查询月份差
select to_char(add_months(trunc(sysdate),-1),'yyyymm') from dual;
- easyui + jdbc 实现简单的数据库管理。
/** * 分页未完成 ,执行sql 查询 与 更新未完成. * 代码拿去,有同学完成了可以再分享出来. * 代码稍微有些乱,没时间整理. * 暂时仅支持oracle ,因sql语句黏合度太高. * ...