这篇文章主要是答复百度网友的问题的,代码做了注释,就不解释了。线程的暂停,我使用了WaitOne方法,线程的继续,则使用了set方法,其实,这两种方法都存在于AutoResetEvent类中,这个类主要是通过设置信号量来实现线程的暂停与继续的。
代码如下:
using System;
using System.Windows.Forms;
using System.Threading;
namespace ThreadInUI
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
for (int i = 0; i < 11; i++)
{
a[i] = new AutoResetEvent(false); //初始化
b[i] = new AutoResetEvent(false);
}
}
string[] threadID = new string[11];
AutoResetEvent[] a = new AutoResetEvent[11];
AutoResetEvent[] b = new AutoResetEvent[11];
private void btnStart_Click(object sender, EventArgs e)
{
lblLog.Invoke((Action)(() => { lblLog.Text += "\r\n当前主线程ID为:" + Thread.CurrentThread.ManagedThreadId.ToString(); }));
for (int i = 0; i < 10; i++)
{
int j = i; //这个用于避免闭包
Thread t = new Thread(new ThreadStart(() =>
{
a[j].WaitOne(); //这个用于获取子线程的ID
lblLog.Invoke((Action)(() => { lblLog.Text += "\r\n线程"+threadID[j]+"睡眠3s"; }));
Thread.Sleep(3000);
lblLog.Invoke((Action)(() => { lblLog.Text += "\r\n线程" + threadID[j] + "暂停"; }));
b[j].WaitOne(); //这个用于暂停所有子线程的运作
lblLog.Invoke((Action)(() => { lblLog.Text += "\r\n===线程"+threadID[j]+"恢复==="; }));
}));
threadID[i] = t.ManagedThreadId.ToString();
a[j].Set(); //当子线程的ID生成后,这里放行,目的是避免取不到Thread的ID
t.Start();
}
}
private void btnGoOn_Click(object sender, EventArgs e)
{
for (int i = 0; i < 10; i++)
{
b[i].Set(); //放行所有的子线程
}
}
}
}
实现的效果如下:
当点击“打开10个线程并阻塞”按钮的时候:
当点击“继续运行10个线程”按钮的时候:
代码很简单,源码我就不加上了。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
2010-10-11 asp.net异步获取datatable并显示