设为首页 收藏本站
查看: 1145|回复: 0

重新想象 Windows 8 Store Apps (43)

[复制链接]

尚未签到

发表于 2015-5-22 05:28:21 | 显示全部楼层 |阅读模式
  [源码下载]




重新想象 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














  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(3000);
// 收到取消操作的请求后抛出一个 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 handler = delegate(object state) // state 是传递过来的上下文对象
            {
// 在当前线程上阻塞 3000 毫秒(当收到取消请求时会发出信号,停止阻塞)
_cts.Token.WaitHandle.WaitOne(3000);
// 收到取消操作的请求后抛出一个 OperationCanceledException 异常,其会导致 task.IsCanceled 的值变为 true
// 此处的代码等同于 _cts.Token.ThrowIfCancellationRequested();
if (_cts.IsCancellationRequested)
throw new OperationCanceledException(_cts.Token);
return "我是“执行一个带返回值的任务”的返回值";
};
// Task.Factory.StartNew() - 创建任务并马上执行,可随时通过 task.Status 获取任务状态
// Task.Run() 同样是创建任务并马上执行
Task task = Task.Factory.StartNew(
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











  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 = 0;
public MultiTask()
{
this.InitializeComponent();
}
private void btnCreateMultiTask_Click_1(object sender, RoutedEventArgs e)
{
// 创建并执行任务1
Task task1 = Task.Run(
() =>
{
new System.Threading.ManualResetEvent(false).WaitOne(3000);
System.Threading.Interlocked.Increment(ref _count);
});
// 创建并执行任务2
Task task2 = Task.Run(
() =>
{
new System.Threading.ManualResetEvent(false).WaitOne(3000);
System.Threading.Interlocked.Increment(ref _count);
});
// 创建并执行任务3
Task task3 = Task.Run(
() =>
{
new System.Threading.ManualResetEvent(false).WaitOne(3000);
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













  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.iyunv.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 = 10 // 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 = 0;
// 未被取消则一直运行
while (!token.IsCancellationRequested)
{
count++;
var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
() =>
{
lblMsg1.Text = "count1: " + count.ToString();
});
token.WaitHandle.WaitOne(100);
}
},
token);
}
// Action 2
private void Task2(CancellationToken token)
{
Task task = Task.Factory.StartNew(
() =>
{
int count = 0;
// 未被取消则一直运行
while (!token.IsCancellationRequested)
{
count++;
var ignored = Dispatcher.RunAsync(Windows.UI.Core.CoreDispatcherPriority.High,
() =>
{
lblMsg2.Text = "count2: " + count.ToString();
});
token.WaitHandle.WaitOne(100);
}
},
token);
}
// 取消并行运算
private void btnCancelParallel_Click_1(object sender, RoutedEventArgs e)
{
if (_cts != null)
_cts.Cancel();
}
}
}
  
OK
[源码下载]

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-69293-1-1.html 上篇帖子: 重新想象 Windows 8 Store Apps (50) 下篇帖子: 重新想象 Windows 8 Store Apps (19)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表