当前位置: 首页 > ai >正文

C#多线程访问资源

文章目录

  • C#多线程访问资源
      • 1.1 锁机制
      • 1.2 信号量机制
      • 1.3 事件与信号
      • 1.4 原子操作
      • 1.5 线程安全集合
      • 1.6 避免共享状态
      • 1.7 异步与并行
      • 1.8 选择策略

C#多线程访问资源

在 C# 中,多线程访问共享资源需要通过同步机制来保证线程安全。以下是常见的解决方案及其适用场景:

1.1 锁机制

  1. lock 关键字

    • 基于 Monitor 类,确保代码块同一时间仅一个线程进入。
    • 适用场景:简单临界区保护。
    private readonly object _lockObj = new object();
    lock (_lockObj) 
    {// 访问共享资源
    }
    
  2. Monitor

    • 提供更灵活的控制(如超时机制)。
    Monitor.Enter(_lockObj);
    try { /* 操作资源 */ }
    finally { Monitor.Exit(_lockObj); }
    
  3. Mutex (互斥锁)

    • 跨进程同步,适用于多进程共享资源。
    using var mutex = new Mutex(false, "GlobalMutexName");
    mutex.WaitOne();
    try { /* 操作资源 */ }
    finally { mutex.ReleaseMutex(); }
    
  4. SpinLock (自旋锁)

    • 通过循环等待避免上下文切换,适用于极短临界区。
    SpinLock spinLock = new SpinLock();
    bool lockTaken = false;
    spinLock.Enter(ref lockTaken);
    try { /* 操作资源 */ }
    finally { if (lockTaken) spinLock.Exit(); }
    

1.2 信号量机制

  1. Semaphore / SemaphoreSlim

    • 控制同时访问资源的线程数量。
    SemaphoreSlim semaphore = new SemaphoreSlim(3); // 允许3个线程进入
    await semaphore.WaitAsync();
    try { /* 操作资源 */ }
    finally { semaphore.Release(); }
    
  2. ReaderWriterLockSlim

    • 读写分离锁,允许多读单写。
    ReaderWriterLockSlim rwLock = new ReaderWriterLockSlim();
    rwLock.EnterReadLock();  // 读模式
    try { /* 读取资源 */ }
    finally { rwLock.ExitReadLock(); }rwLock.EnterWriteLock(); // 写模式
    try { /* 修改资源 */ }
    finally { rwLock.ExitWriteLock(); }
    

1.3 事件与信号

  1. ManualResetEvent / AutoResetEvent

    • 通过信号控制线程阻塞与唤醒。
    ManualResetEvent mre = new ManualResetEvent(false);
    mre.WaitOne();    // 等待信号
    mre.Set();        // 发送信号
    
  2. Barrier (屏障)

    • 同步多个线程到同一阶段。
    Barrier barrier = new Barrier(3); // 等待3个线程到达
    barrier.SignalAndWait();          // 每个线程调用此方法
    
  3. CountdownEvent

    • 等待指定数量的操作完成。
    CountdownEvent cde = new CountdownEvent(3);
    cde.Signal();  // 每个线程完成后调用
    cde.Wait();    // 等待所有完成
    

1.4 原子操作

  1. Interlocked

    • 提供原子操作(如递增、比较交换)。
    int value = 0;
    Interlocked.Increment(ref value); // 原子递增
    

1.5 线程安全集合

  1. ConcurrentQueue / ConcurrentDictionary

    • 内置线程安全的集合,避免手动同步。
    var queue = new ConcurrentQueue<int>();
    queue.Enqueue(1);
    if (queue.TryDequeue(out var item)) { /* 处理元素 */ }
    

1.6 避免共享状态

  1. 不可变对象

    • 使用 readonly 或不可变集合(如 ImmutableList),确保数据不可变。
    var list = ImmutableList.Create<int>();
    list = list.Add(1); // 返回新实例,原数据不变
    
  2. 线程本地存储

    • 使用 ThreadLocal<T>[ThreadStatic] 为每个线程创建独立副本。
    ThreadLocal<int> threadLocal = new ThreadLocal<int>(() => 0);
    int localValue = threadLocal.Value;
    

1.7 异步与并行

  1. async/await 与异步锁

    • 在异步代码中使用 SemaphoreSlim.WaitAsync()
    private SemaphoreSlim asyncLock = new SemaphoreSlim(1);
    await asyncLock.WaitAsync();
    try { /* 异步操作资源 */ }
    finally { asyncLock.Release(); }
    
  2. TPL (任务并行库)

    • 使用 Parallel.ForTask 时确保资源安全。
    Parallel.For(0, 10, i => 
    {// 需要内部同步机制
    });
    

1.8 选择策略

  • 简单临界区:优先使用 lockMonitor
  • 读写分离:使用 ReaderWriterLockSlim
  • 高并发读:不可变对象或并发集合。
  • 异步场景SemaphoreSlim.WaitAsync()
  • 跨进程同步Mutex

通过合理选择同步机制,可以平衡性能与线程安全。

http://www.xdnf.cn/news/1287.html

相关文章:

  • Node.js 开发用户登录功能(使用mysql实现)
  • 【AI应用】免费代码仓构建定制版本的ComfyUI应用镜像
  • 【Linux应用】RADXA ZERO 3快速上手:镜像烧录、串口shell、外设挂载、WiFi配置、SSH连接、文件交互
  • Zookeeper是什么?基于zookeeper实现分布式锁
  • 软件黑盒与白盒测试详解
  • 同样的接口用postman/apifox能跑通,用jmeter跑就报错500
  • 【MCP】第二篇:IDE革命——用MCP构建下一代智能工具链
  • 【Linux】冯诺依曼体系结构及操作系统架构图的具体剖析
  • 【Ubuntu】关于系统分区、挂载点、安装位置的一些基本信息
  • 【算法笔记】动态规划基础(一):dp思想、基础线性dp
  • 【k8s】docker、k8s、虚拟机的区别以及使用场景
  • sentinel
  • CATBOOST算法总结
  • vscode如何多行同时编辑,vscode快速选中多行快捷键
  • 使用 JUnit 4在 Spring 中进行单元测试的完整步骤
  • 【数据结构入门训练DAY-21】信息学奥赛一本通T1334-围圈报数
  • 深入剖析TCP协议(内容二):从OSI与TCP/IP网络模型到三次握手、四次挥手、状态管理、性能优化及Linux内核源码实现的全面技术指南
  • 基于cubeMX的hal库STM32实现MQ2烟雾浓度检测
  • 软考软件设计师30天备考指南
  • 升级xcode16之后react-native-zip-archive不兼容,unsupported option ‘-G‘
  • The backpropagation and the brain
  • Java与C语言核心差异:从指针到内存管理的全面剖析
  • Node.js学习
  • WT2000T专业录音芯片:破解普通录音设备信息留存、合规安全与远程协作三大难题
  • 【k8s系列7-更新中】kubeadm搭建Kubernetes高可用集群-三主两从
  • .NET 6 WPF 利用CefSharp.Wpf.NETCore显示PDF文件
  • 什么是 GLTF/GLB? 3D 内容创建的基本数据格式说明,怎么下载GLB/GLTF格式模型
  • HarmonyOS 是 Android 套壳嘛?
  • 【C语言】动态内存的常见错误
  • Git远程操作与标签管理