C#任务并行库TPL--Task应用

一、概念

  TPL的核心就是任务,一个任务代表一个异步操作,该操作可以通过多种方式运行,一个任务也可以由多个任务组成。

二、应用

1、创建任务有三种方法:  

复制代码
var t1 = new Task(() => TaskMethod("Task 1"));                //通过Task构造函数创建
var t2 = new Task(() => TaskMethod("Task 2"));
t2.Start();                                     //需要手动启动
t1.Start();                                    
Task.Run(() => TaskMethod("Task 3"));                     //Task.Run()可以自启动
Task.Factory.StartNew(() => TaskMethod("Task 4"));             //Task.Factory.StartNew()相对于前者要可以添加附加选项,如下添加了标记任务为长时间运行
Task.Factory.StartNew(() => TaskMethod("Task 5"),TaskCreationOptions.LongRunning);
Sleep(TimeSpan.FromSeconds(1));
Console.ReadKey();
复制代码

 

 创建的任务都会被放入线程池,而且没有指定顺序。Task 5 经过标记后,不会使用线程池。

2、基本操作

Task<int> task = new Task<int>(()=>TaskMethod(name));

task.RunSynchronously();
//通过该方法运行的任务会运行在主线程上,可以避免使用线程池来执行非常短暂的操作
firstTask.ContinueWith(t => WriteLine(),TaskContinuationOptions.OnlyOnRanToCompletion);
// 该代码表示当firstTask这个任务的一个后续操作,在当前任务完成后运行

Task continuation = secondTask.ContinueWith(t => WriteLine(...,
     TaskContinuationOptions.OnlyOnRanToCompletion| TaskContinuationOptions.ExecuteSynchronously);
//TaskContinuationOptions.OnlyOnRanToCompletion| TaskContinuationOptions.ExecuteSynchronously
//用来尝试同步后续操作

创建子任务:

复制代码
firstTask = new Task<int>(() =>
            {
                var innerTask = Task.Factory.StartNew(() => TaskMethod(
                    "Second Task", 5), TaskCreationOptions.AttachedToParent);
                innerTask.ContinueWith(t => TaskMethod("Third Task", 2),
                    TaskContinuationOptions.AttachedToParent);
                return TaskMethod("First Task", 2);
            });
            //上述TaskCreationOptions.AttachedToParent运行了一个子任务
            //通过TaskContinuationOptions.AttachedToParent给子任务一个后续操作,该操作会影响到父任务
            //因为父任务必须等到所有的子任务结束它才会运行完
复制代码

取消操作:

复制代码
        var cts = new CancellationTokenSource();
            var longTask = new Task<int>(() => TaskMethod("Task 1", 10, cts.Token), cts.Token);
            WriteLine(longTask.Status);
            cts.Cancel();                                      //此时状态为Canceled
            //longTask.Start();
            WriteLine(longTask.Status);
            WriteLine("First task has been cancelled before execution");

            cts = new CancellationTokenSource();
            longTask = new Task<int>(() => TaskMethod("Task 2", 10, cts.Token), cts.Token);
            longTask.Start();
            for (int i = 0; i < 5; i++)
            {
                Sleep(TimeSpan.FromSeconds(0.5));
                WriteLine(longTask.Status);
            }
            cts.Cancel();                                      //此时任务的状态变为RanToCompletion
            //longTask.Start();
            for (int i = 0; i < 5; i++)
            {
                Sleep(TimeSpan.FromSeconds(0.5));
                WriteLine(longTask.Status);
            }
复制代码

三、注意事项

1、取消任务的时候,任务所在的线程需要一定的时间把这个消息传递给主线程,在主线程收到消息之前,都会认为这个任务还是原来状态。

2、不要混淆任务跟线程的概念。线程是最小的单位,而一个任务可以由多个任务组成。

 

posted @   Darius丶段  阅读(226)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· .NET10 - 预览版1新功能体验(一)
点击右上角即可分享
微信分享提示