C#中异步编程时使用Task.Factory.StartNew和Task.Run的区别
摘要
在C#中做异步编程是,我们习惯使用 Task.Factory.StartNew和Task.Run,那么这两个方法有什么区别呢?用哪一个更好呢?
下面就来给大家分析分析。
区别
- Task.Run 是在 .Net Framework 4.5 之后才提供的方法
- Task.Factory.StartNew 可以使用比 Task.Run 更多的参数,可以做到更多的任务设置
- 可以认为 Task.Run 是简化的 Task.Factory.StartNew 的使用,除了需要指定一个线程是长时间占用的,否则就使用 Task.Run
创建新线程
接下来,给大家举例,使用两个函数创建新的线程。
Task.Run(() =
{
var v1 = 2;
});
这时 v1 的创建就在另外的线程中,Task.Run 用的是线程池,也就是不是调用这个函数就会一定创建一个新的线程,但是会在另一个线程运行。关于线程池就不多在这里介绍了。
Task.Factory.StartNew(() =
{
var v2 = 2;
});
可以看到,两个方法实际上是没有差别,但是Task.Runs书写更为简洁,所以推荐使用Task.Run方法。
异步等待
在异步编程时,创建的线程,如果需要等待线程执行完成再继续,那么可以使用 await 等待
private static async void SeenereKousa()
{
Console.WriteLine(开始 线程+Thread.CurrentThread.ManagedThreadId);
await Task.Run(() =
{
Console.WriteLine(进入 线程 + Thread.CurrentThread.ManagedThreadId);
});
Console.WriteLine(退出 线程+Thread.CurrentThread.ManagedThreadId);
}
但是需要说明的是,这里使用 await 主要是给Task后续的代码使用,上面代码在函数里面使用 await 函数返回值是void 那么和把代码放在 task 里面执行结果是一样的。
private static async void SeenereKousa()
{
Console.WriteLine(开始 线程+Thread.CurrentThread.ManagedThreadId);
await Task.Run(() =
{
Console.WriteLine(进入 线程 + Thread.CurrentThread.ManagedThreadId);
Console.WriteLine(退出 线程+Thread.CurrentThread.ManagedThreadId);
});
}
但是如果把 void 修改为 Task ,那么等待线程才有用。
除了使用 await 等待,还可以使用 WaitAll 等待
Console.WriteLine(开始 线程 + Thread.CurrentThread.ManagedThreadId);
var t = Task.Run(() =
{
Console.WriteLine(进入 线程 + Thread.CurrentThread.ManagedThreadId);
});
Task.WaitAll(t);
Console.WriteLine(退出 线程 + Thread.CurrentThread.ManagedThreadId);
使用 WaitAll 方法是在调用 WaitAll 的线程中等待,也就是先在线程 1 运行,然后异步到 线程2 运行,这时线程1 等待线程2运行完成再继续,所以输出结果如下:
开始 线程1
进入 线程2
退出 线程1
长线程
长线程,顾名思义就是需要较长时间保持运行的线程。
两个函数最大的不同在于 Task.Factory.StartNew 可以将线程设置为长线程,这时线程池就不会等待这个线程回收。
Task.Factory.StartNew(() =
{
for (int i = 0; i 100; i++)
{
var v1 = 2;
}
Console.WriteLine(进行 线程 + Thread.CurrentThread.ManagedThreadId);
}, TaskCreationOptions.LongRunning);
结论
所以在需要设置线程是长时间运行的才需要使用 Task.Factory.StartNew 不然就使用 Task.Run
调用 Task.Run(fun) 就和使用下面代码一样
Task.Factory.StartNew(foo,
CancellationToken.None, TaskCreationOptions.DenyChildAttach, TaskScheduler.Default);
实际上 Task.Run(fun) 可以认为是对 Task.Factory.StartNew 封装,使用简单的默认的参数。如果需要自己定义很多参数,就请使用 Task.Factory.StartNew 定义参数。
更新于:18天前相关文章
- composer install、update、require 区别
- .NET中Task和ValueTask的区别
- PHP 函数传值和传引用的区别
- require 和 include 的区别
- Navicat Premium 和 Navicat for MySQL 的区别
- .NET8 Blazor三种模式的区别和使用场景
- C++释放new分配内存时带方括号delete[]和不带方括号delete的区别
- Shadowsocks各个加密算法的区别
- 程序员和码农的区别:解析软件开发领域的两种角色
- java中get请求和post请求的区别
- Makefile中赋值等号:=, =, ?=和+=的含义和区别
- .NET奇数和偶数版本的区别
- .NET7和.NET8的区别
- ASP.NET MVC和ASP.NET Core的区别是什么?
- SVN和Git的区别?
- localhost与127.0.0.1的区别
- QT编译时报错 error: lnk1158: cannot run 'rc.exe'
- H5和html的区别
- ipv4和ipv6的区别
- MongoDB和Redis的区别?