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

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
简介: 原文:重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)[源码下载] 重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(P...
原文: 重新想象 Windows 8 Store Apps (43) - 多线程之任务: Task 基础, 多任务并行执行, 并行运算(Parallel)

[源码下载]


重新想象 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(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<object, string> 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<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 = 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

<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 = 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
[源码下载]

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
4月前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android多线程编程的重要性及其实现方法,涵盖了基本概念、常见线程类型(如主线程、工作线程)以及多种多线程实现方式(如`Thread`、`HandlerThread`、`Executors`、Kotlin协程等)。通过合理的多线程管理,可大幅提升应用性能和用户体验。
155 15
一个Android App最少有几个线程?实现多线程的方式有哪些?
|
4月前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android应用开发中的多线程编程,涵盖基本概念、常见实现方式及最佳实践。主要内容包括主线程与工作线程的作用、多线程的多种实现方法(如 `Thread`、`HandlerThread`、`Executors` 和 Kotlin 协程),以及如何避免内存泄漏和合理使用线程池。通过有效的多线程管理,可以显著提升应用性能和用户体验。
123 10
|
5月前
|
Windows Python
【Azure 批处理 Azure Batch】在Azure Batch中如何通过开始任务自动安装第三方依赖的一些软件(Windows环境)
【Azure 批处理 Azure Batch】在Azure Batch中如何通过开始任务自动安装第三方依赖的一些软件(Windows环境)
|
5月前
|
Windows
Windows——如何提取Microsoft Store的应用
Windows——如何提取Microsoft Store的应用
40 0
|
7月前
|
网络虚拟化 Windows
Windows 10 Windows1011出现0x80190001错误解决方案! Windows微软账户无法登录问题 Microsoft Store商店用不了
Windows 10 Windows1011出现0x80190001错误解决方案! Windows微软账户无法登录问题 Microsoft Store商店用不了
170 1
|
6月前
|
JavaScript IDE 持续交付
阿里云云效产品使用合集之如何配置 Node.js构建任务让其在Windows环境中进行
云效作为一款全面覆盖研发全生命周期管理的云端效能平台,致力于帮助企业实现高效协同、敏捷研发和持续交付。本合集收集整理了用户在使用云效过程中遇到的常见问题,问题涉及项目创建与管理、需求规划与迭代、代码托管与版本控制、自动化测试、持续集成与发布等方面。
|
7月前
|
安全 API C++
逆向学习Windows篇:C++中多线程的使用和回调函数的实现
逆向学习Windows篇:C++中多线程的使用和回调函数的实现
233 0
|
8月前
|
Windows
Windows逆向 -- 逻辑运算与位移
Windows逆向 -- 逻辑运算与位移
38 0
|
8月前
|
Linux iOS开发 Windows
windows 如何上架 ios 应用到 app store
windows 如何上架 ios 应用到 app store
|
8月前
|
Shell 数据安全/隐私保护 Windows
Windows Server【开机启动和任务计划程序】实现服务器重启后项目自启动(Windows Server 任务计划程序无法执行问题处理)
Windows Server【开机启动和任务计划程序】实现服务器重启后项目自启动(Windows Server 任务计划程序无法执行问题处理)
601 0