C#基础——多线程Task
C#基础——多线程Task
Task是 System.Threading 提供的多线程类,Task有2种模式启动——start和run,
Task.Run 示例
Task<int> t2 = Task.Run(() =>
{
int res = 0;
for (int i = 0; i < 10; i++) {
Console.WriteLine("t2:" + i );
res += i;
Thread.Sleep(500);
}
return res;
});
Task(Action<>).Start() 示例
new Task(() =>
{
for (int i = 0; i < 10; i++) {
Console.WriteLine("t2:" + i );
res += i;
Thread.Sleep(500);
}
}).Start();
Wait() 阻塞当前线程,等待一个任务执行
Task.Run
Task t = Task.Run(() => {
Console.WriteLine(5);
Thread.Sleep(1000);
});
t.Wait();//等待t任务完成后再接着执行
Console.WriteLine(10);
Console.ReadLine();
task.Start
Task t2 = new Task(() => {
Console.WriteLine(5);
Thread.Sleep(1000);
});
t2.Start();
t2.Wait();
Console.WriteLine(10);
Console.ReadLine();
WaitAll() 阻塞当前线程,等待多任务执行
Task.Run
Task t1 = Task.Run(() => { Console.WriteLine(0); Thread.Sleep(1000); });
Task t2 = Task.Run(() => { Console.WriteLine(1); Thread.Sleep(1000); });
Task t3 = Task.Run(() => { Console.WriteLine(2); Thread.Sleep(1000); });
Task.WaitAll(new Task[] { t1, t2, t3 });// 等待所有任务执行完毕后接着往下执行
Console.WriteLine(10);
task.Start
Task t1 = new Task(() => { Console.WriteLine(0); Thread.Sleep(1000); });
Task t2 = new Task(() => { Console.WriteLine(1); Thread.Sleep(1000); });
Task t3 = new Task(() => { Console.WriteLine(2); Thread.Sleep(1000); });
t1.Start();
t2.Start();
t3.Start();
Task.WaitAll(new Task[] { t1, t2, t3 });// 等待所有任务执行完毕后接着往下执行
超时等待
主线程仅阻塞固定时间,超过此时间段,不管任务是否完成,均继续主线程的任务往下执行
Wait
Task t1 = new Task(() => { Console.WriteLine(0); Thread.Sleep(1000); });
t1.Wait(500);//超时等待500毫秒
Console.WriteLine($"{t1.Status}");// running状态,因为此时t1还没有执行完成
WaitAll
Task t1 = new Task(() => { Console.WriteLine(0); Thread.Sleep(1000); });
Task t2 = new Task(() => { Console.WriteLine(1); Thread.Sleep(1000); });
Task t3 = new Task(() => { Console.WriteLine(2); Thread.Sleep(1000); });
Console.WriteLine($"{t1.Status} {t2.Status} {t3.Status}");
t1.Start();
t2.Start();
t3.Start();
Console.WriteLine($"{t1.Status} {t2.Status} {t3.Status}");
Task.WaitAll(new Task[] { t1, t2, t3 }, 1000);
Console.WriteLine($"{t1.Status} {t2.Status} {t3.Status}");
Status 获取线程状态
Task t1 = new Task(() => { Console.WriteLine(0); Thread.Sleep(1000); });
Task t2 = new Task(() => { Console.WriteLine(1); Thread.Sleep(1000); });
Task t3 = new Task(() => { Console.WriteLine(2); Thread.Sleep(1000); });
Console.WriteLine($"{t1.Status} {t2.Status} {t3.Status}");
t1.Start(); t2.Start(); t3.Start();
Console.WriteLine($"{t1.Status} {t2.Status} {t3.Status}");
Task.WaitAll(new Task[] { t1, t2, t3 });
Console.WriteLine($"{t1.Status} {t2.Status} {t3.Status}");
线程状态:TaskStatus(枚举)
Created = 0 该任务已初始化,但尚未被计划
WaitingForActivation = 1 该任务正在等待 .NET Framework 基础结构在内部将其激活并进行计划
WaitingToRun = 2 该任务已被计划执行,但尚未开始执行
Running = 3 该任务正在运行,但尚未完成
WaitingForChildrenToComplete = 4 该任务已完成执行,正在隐式等待附加的子任务完成
RanToCompletion = 5 已成功完成执行的任务
Canceled = 6 取消任务
Faulted = 7 由于未处理异常的原因而完成的任务
获取任务返回值
仅Task.Run支持
Task<int> t2 = Task.Run(() =>
{
int res = 0;
for (int i = 0; i < 10; i++) {
Console.WriteLine("t2:" + i );
res += i;
Thread.Sleep(500);
}
return res;
});
int resault = t2.Result;// 阻塞主线程,等待t2线程完成
后任务
完成了A任务后,需要做一些收尾工作,或者完成A任务后,B任务才能开始做。
Task<int> t1 = Task.Run(() => {
int res = 0;
for (int i = 1; i <= 4; i++)
res += i;
return res;
});
// 后续任务,接着t1执行完成后执行
Task<string> continueT1 = t1.ContinueWith((t) => {
if (t.Status == TaskStatus.RanToCompletion)
{
Console.WriteLine($"result:{t.Result}");
return "success";
}
else
{
return "faild";
}
});
// 紧接continueT1任务执行完成后执行
Task finishTask = continueT1.ContinueWith((t) => {
if (t.Status == TaskStatus.RanToCompletion)
{
Console.WriteLine($"finish:{t.Result}");
}
});
Thread.Sleep(1000);// 主线程休眠10秒
Console.ReadLine();
Task(Action<>).Start() 和 Task.Run 的区别
- Task.Start的lambda表达式仅支持Action,Task.run支持Action和Func
Task.Run() 支持返回值,Task().Start()不支持
Task.Start()返回值是void,Task.Run() 返回值是Task
- Task.Run() 更利于管理,可以更方便地用Task.Status获取线程状态,而Task().Start()不行
- Task.Run 支持捕获异常,但Task().Start()可能不会捕获到异常
官方更加推荐使用Task.Run执行任务
本文来自博客园,作者:勤匠,转载请注明原文链接:https://www.cnblogs.com/JarryShu/articles/18190994
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现