在系统中单开线程进行操作,经常用到Task,发现Task主要有以下两种方法
Task.Factory.StartNew(() => { });
Task.Run(() => { });
初识不知其中区别,特意参考其他文章已做总结https://www.mgenware.com/blog/?p=338
首先说明Task.UnWrap的基本使用
这个扩展方法定义在TaskExtensions类型中,命名空间在System.Threading.Tasks。Unwrap会把嵌套的Task<Task>或者Task<Task<T>>的结果提取出来。
就像这样,不用Unwrap的话:
static void Main(string[] args)
{
doo();
Task.Delay(-).Wait();
} static async void doo()
{
//运行嵌套的Task
//Task返回Task<Task<string>>
//第一个await后result类型为Task<string>
var result = await Task.Run<Task<string>>(() =>
{
var task = Task.Run<string>(() =>
{
Task.Delay().Wait();
return "Mgen";
});
return task;
}); //第二个await后才会返回string
Console.WriteLine(await result);
}
使用Unwrap后,结果可以直接从嵌套Task中提取出来:
static async void doo()
{
//运行嵌套的Task
//Task返回Task<Task<string>>
//await后类型为Task<string>,Unwrap后result类型为string
var result = await Task.Run<Task<string>>(() =>
{
var task = Task.Run<string>(() =>
{
Task.Delay().Wait();
return "Mgen";
});
return task;
}).Unwrap(); //不需要await,result已经是string
Console.WriteLine(result);
}
简单地讲,Task.Factory.StartNew和Task.Run区别之一就有Task.Run会自动执行Unwrap操作,但是Task.Factory.StartNew不会,Task.Run就是Task.Factory.StartNew的更人性化封装,而Task.Factory.StartNew则是原始的执行。
通过代码来验证:
var task1 = Task.Factory.StartNew(async () => "Mgen");
var task2 = Task.Run(async () => "Mgen"); Console.WriteLine(task1.GetType());
Console.WriteLine(task2.GetType());
输出
System.Threading.Tasks.Task`[System.Threading.Tasks.Task`[System.String]]
System.Threading.Tasks.UnwrapPromise`[System.String]
可以看到
使用Task.Factory.StartNew会返回原始的Task<Task<string>>。但是Task.Run则会直接返回async Lambda的结果,中间的Unwrap操作会自动进行。