C#中的ThreadStart委托

ThreadStart 委托
ThreadStart 是 .NET 中的一个内置委托类型,表示无参数且无返回值的方法。其定义如下:

cs 复制代码
public delegate void ThreadStart();
  • 通常用于定义线程的入口方法。

  • List<ThreadStart>

    这是一个泛型集合,用于存储多个 ThreadStart 委托实例。每个委托代表一个待执行的任务。

  • 整体作用

    创建一个线程任务队列,用于保存需要通过线程执行的多个方法。

2. 典型使用场景

cs 复制代码
private List<ThreadStart> delegates = new List<ThreadStart>();
(1) 多任务队列管理
cs 复制代码
// 添加任务到队列
delegates.Add(() => Console.WriteLine("Task 1"));
delegates.Add(() => File.WriteAllText("test.txt", "Hello"));

// 启动线程执行所有任务
foreach (var task in delegates)
{
    new Thread(task).Start();
}

(2) 延迟执行控制

cs 复制代码
// 先收集任务
delegates.Add(() => DownloadFile(url1));
delegates.Add(() => ProcessData(data));

// 在适当时候触发执行
void ExecuteAllTasks()
{
    foreach (var task in delegates)
    {
        new Thread(task).Start();
    }
}

3. 技术细节

委托与线程的关系
  • 每个 ThreadStart 委托可以传递给 Thread 构造函数,作为线程启动时执行的方法。

  • 示例:

cs 复制代码
ThreadStart task = () => Console.WriteLine("Running in thread");
Thread thread = new Thread(task);
thread.Start();
线程安全注意事项
  • 非线程安全集合
    List<T> 本身不是线程安全的。若多线程同时修改集合(如添加/删除任务),需加锁:
cs 复制代码
private readonly object _lock = new object();

void AddTask(ThreadStart task)
{
    lock (_lock)
    {
        delegates.Add(task);
    }
}

4. 完整使用示例

cs 复制代码
using System;
using System.Collections.Generic;
using System.Threading;

class TaskScheduler
{
    private List<ThreadStart> _tasks = new List<ThreadStart>();
    private readonly object _lock = new object();

    public void AddTask(Action action)
    {
        lock (_lock)
        {
            _tasks.Add(new ThreadStart(action));
        }
    }

    public void ExecuteAll()
    {
        List<Thread> threads = new List<Thread>();
        
        lock (_lock)
        {
            foreach (var task in _tasks)
            {
                Thread thread = new Thread(task);
                threads.Add(thread);
                thread.Start();
            }
            _tasks.Clear();
        }

        // 等待所有线程完成(可选)
        foreach (var thread in threads)
        {
            thread.Join();
        }
    }
}

// 使用示例
var scheduler = new TaskScheduler();
scheduler.AddTask(() => Console.WriteLine("Task 1"));
scheduler.AddTask(() => Thread.Sleep(1000));
scheduler.ExecuteAll();

5. 替代方案(现代C#推荐)

使用 TaskConcurrentQueue
cs 复制代码
using System.Collections.Concurrent;
using System.Threading.Tasks;

private ConcurrentQueue<Action> _taskQueue = new ConcurrentQueue<Action>();

// 添加任务
_taskQueue.Enqueue(() => Console.WriteLine("Task 1"));

// 并行执行
Parallel.ForEach(_taskQueue, task => task.Invoke());
_taskQueue.Clear();
优点
  • 更高效的线程池管理(通过 Task

  • 天生线程安全的集合(ConcurrentQueue

  • 支持 async/await


6. 关键区别:ThreadStart vs Action

特性 ThreadStart Action
返回值 无 (void) 无 (void)
参数 可带参数(如 Action<int>
用途 专用于 Thread 构造函数 通用委托
现代性 较旧 API 推荐使用

总结

  • 原始代码 :创建了一个传统的线程任务队列,适用于需要显式管理 Thread 的场景。

  • 现代替代 :推荐使用 Task + ConcurrentQueue 组合,更符合当前 .NET 的并发编程最佳实践。

  • 线程安全 :若坚持使用 List<ThreadStart>,必须通过锁机制保证线程安全。

根据实际需求选择合适方案,平衡控制精细度和开发效率。

相关推荐
CallZhang2102 小时前
Vision Master的C#脚本与opencv联合编程
opencv·计算机视觉·c#·视觉检测
AI视觉网奇2 小时前
kafka 冲突解决 kafka安装
c#·linq
hqwest2 小时前
C#WPF实战出真汁07--【系统设置】--菜品类型设置
开发语言·c#·wpf·grid设计·stackpanel布局
萘柰奈3 小时前
Unity进阶--C#补充知识点--【Unity跨平台的原理】Mono与IL2CPP
unity·c#·游戏引擎
程序设计实验室3 小时前
StarBlog v1.3.0 新版本,一大波更新以及迁移服务器部署
c#·aspnetcore·starblog番外
淡海水4 小时前
【原理】Struct 和 Class 辨析
开发语言·c++·c#·struct·class
淡海水4 小时前
【原理】Unity GC 对比 C# GC
unity·c#·gc·垃圾回收
张人玉5 小时前
C#读取文件, IO 类属性及使用示例
microsoft·c#
咕白m62510 小时前
通过 C# 高效提取 PDF 文本的完整指南
后端·c#
hqwest13 小时前
C#WPF实战出真汁08--【消费开单】--餐桌面板展示
c#·wpf·ui设计·wpf界面设计