经验大分享:Task的用法

简介: 经验大分享:Task的用法

Task用法记录总结

1.创建Task的3种方法

1.1new方式实例化一个Task,需要通过Start方法启动

Task task = new Task(() =>

{

Thread.Sleep(100);

Console.WriteLine($"hello, task1的线程ID为{Thread.CurrentThread.ManagedThreadId}");

});

task.Start();

task.Start方式,每个task都在不同的线程中

代码:

private static void Main(string【】 args)

{

Console.WriteLine("Hello, World!");

Task task1 = new Task(() =>

{

Console.WriteLine("task1 is running on a thread id {0}. Is thread pool thread: {1}",

Thread.CurrentThread.ManagedThreadId,

Thread.CurrentThread.IsThreadPoolThread);

});

Task task2 = new Task(() =>

{

Console.WriteLine("task2 is running on a thread id {0}. Is thread pool thread: {1}",

Thread.CurrentThread.ManagedThreadId,

Thread.CurrentThread.IsThreadPoolThread);

});

Task task3 = new Task(() =>

{

Console.WriteLine("task3 is running on a thread id {0}. Is thread pool thread: {1}",

Thread.CurrentThread.ManagedThreadId,

Thread.CurrentThread.IsThreadPoolThread);

});

//不在同一个线程

task1.Start();

task2.Start();

task3.Start();

//在同一个线程

//task1.RunSynchronously();

//task2.RunSynchronously();

//task3.RunSynchronously();

Task.WaitAll(task1,task2,task3);

}

View Code

task.RunSynchronously方式,每个任务都在同一个线程。task会阻塞主线程

task.Start方式,不会阻塞主线程

1.2Task.Factory.StartNew(Action action)创建和启动一个Task

Task task2 = Task.Factory.StartNew(() =>

{

Thread.Sleep(100);

Console.WriteLine($"hello, task2的线程ID为{Thread.CurrentThread.ManagedThreadId}");

});

创建后自动运行Task

1.3Task.Run(Action action)将任务放在线程池队列,返回并启动一个Task

Task task3 = Task.Run(() =>

{

Thread.Sleep(100);

Console.WriteLine($"hello, task3的线程ID为{Thread.CurrentThread.ManagedThreadId}");

});

创建后自动运行Task

2.Task的Wait/WaitAny/WaitAll方法

task.Wait() 表示等待task执行完毕

Task.WaitAll(Task【】 tasks) 表示只有所有的task都执行完成了再解除阻塞

private static void Main(string【】 args)

{

Console.WriteLine("start");

Task task1 = new Task(() =>

{

Thread.Sleep(1000);

Console.WriteLine("task1 is running on a thread id {0}//代码效果参考:http://www.ezhiqi.com/zx/art_3426.html . Is thread pool thread: {1}",

Thread.CurrentThread.ManagedThreadId,

Thread.CurrentThread.IsThreadPoolThread);

});

Task task2 = new Task(() =>

{

Thread.Sleep(2000);

Console.WriteLine("task2 is running on a thread id {0}. Is thread pool thread: {1}",

Thread.CurrentThread.ManagedThreadId,

Thread.CurrentThread.IsThreadPoolThread);

});

Task task3 = new Task(() =>

{

Thread.Sleep(3000);

Console.WriteLine("task3 is running on a thread id {0}. Is thread pool thread: {1}",

Thread.CurrentThread.ManagedThreadId,

Thread.CurrentThread.IsThreadPoolThread);

});

// 不在同一个线程

task1.Start();

task2.Start();

task3.Start();

Task.WaitAll(task1,task2,task3);

Console.WriteLine("task 1 2 3 end");

}

task1 等待1秒,task2等待2秒 task3等待3秒,使用WaitAll,则会等所有任务执行完毕之后才结束。

Task.WaitAny(Task【】 tasks) 表示只要有一个task执行完毕就解除阻塞

task1 等待1秒,task2等待2秒 ,task3等待3秒,使用WaitAny,只要有一个task结束,就结束阻塞。这里task最先结束

3.Task的延续操作(WhenAny/WhenAll/ContinueWith)

Task.WhenAll(task1, task2, task3).ContinueWith((t) => {

//dosomething

Console.WriteLine("task 1 2 3 all End");

});

当task完成后执行某些操作。

4.带返回值的Task

private static int Main(string【】 args)

{

//1.new方式实例化一个Task

Task[span style="color: rgba(0, 0, 255, 1)">intint

{

Thread.Sleep(100);

Console.WriteLine($"hello, task1的线程ID为{Thread.CurrentThread.ManagedThreadId}");

return 12;

});

task.Start();

int ret = task.Result;

task.Wait();

Console.WriteLine(ret);

//2.Task.Run

Task[span style="color: rgba(0, 0, 255, 1)">intint

{

Thread.Sleep(1000);

Console.WriteLine($"hello, task2的线程ID为{Thread.CurrentThread.ManagedThreadId}");

return 13;

});

int ret2 = task2.Result;

Console.WriteLine(ret2);

//3.Task.Factory.StartNew

Task[span style="color: rgba(0, 0, 255, 1)">int

{

Thread.Sleep(100);

Console.WriteLine($"hello, task3的线程ID为{Thread.CurrentThread.ManagedThreadId}");

return 14;

});

int ret3 = task3.Result;

Console.WriteLine(ret3);

return 0;

}

5.异步async awit

private static void Main(string【】 args)

{

Console.WriteLine("begin download");

Task task = download();

Console.WriteLine("end download");

}

static async Task[span style="color: rgba(0, 0, 255, 1)">int

{

Console.WriteLine("开始");

await Task.Run(() => { Thread.Sleep(2222); });

Console.WriteLine("结束");

return 666;

}

可以看到主线程退出了,但download异步方法还没结束。

深入理解异步:

一文说通C#中的异步编程 - 老王Plus - 博客园 (cnblogs.com)

相关文章
|
6月前
|
存储
向量化代码实践问题之Task<T>类中的on_completed函数是如何工作的
向量化代码实践问题之Task<T>类中的on_completed函数是如何工作的
|
8月前
|
Java 调度
一张图搞清楚wait、sleep、join、yield四者区别,面试官直接被征服!
一张图搞清楚wait、sleep、join、yield四者区别,面试官直接被征服!
61 2
|
8月前
|
Java C#
C#学习相关系列之多线程(七)---Task的相关属性用法
C#学习相关系列之多线程(七)---Task的相关属性用法
|
前端开发
前端学习案例1-call,apply的重用
前端学习案例1-call,apply的重用
83 0
前端学习案例1-call,apply的重用
高并发编程-深入分析wait和sleep的区别并结合源码示例佐证
高并发编程-深入分析wait和sleep的区别并结合源码示例佐证
122 0
ts重点学习126-装饰器的小例子笔记
ts重点学习126-装饰器的小例子笔记
95 0
ts重点学习52-构造函数笔记
ts重点学习52-构造函数笔记
114 0
ts重点学习119-方法的装饰器笔记
ts重点学习119-方法的装饰器笔记
70 0
|
机器学习/深度学习 Python
Python语言学习:复杂函数(yield/@property)使用方法、案例应用之详细攻略
Python语言学习:复杂函数(yield/@property)使用方法、案例应用之详细攻略
|
Java 测试技术
Task 异步编程测试案例及基础应用说明
 对于多线程,我们经常使用的是Thread。在我们了解Task之前,如果我们要使用多核的功能可能就会自己来开线程,然而这种线程模型在.net 4.0之后被一种称为基于“任务的编程模型”所冲击,因为task会比thread具有更小的性能开销,不过大家肯定会有疑惑,任务和线程到底有什么区别呢?  任务和线程的区别: 1、任务是架构在线程之上的,也就是说任务最终还是要抛给线程去执行。
1108 0