C#中实现跨线程写入的示例代码

 更新时间:2026年01月09日 11:26:31   作者:就是有点傻  
本文主要介绍了C#中实现跨线程写入的示例代码,通过ConcurrentQueue接收写入请求,由专用线程顺序处理,并使用双信号机制控制读取线程的暂停与恢复,下面就来详细的介绍一下

方案核心思路

  • 写入请求队列:使用 ConcurrentQueue 接收来自任意线程的写入请求。
  • 专用写入线程:由独立线程处理队列中的写入操作,确保顺序执行。
  • 双信号机制:通过 ManualResetEventSlim 控制读取线程的暂停与恢复。
  • 线程安全确认:确保多个线程同时触发写入时,不会导致竞态条件。

完整代码实现

using System;
using System.Collections.Concurrent;
using System.Threading;

public class CrossThreadReadWriteController
{
    // 控制读取线程暂停和恢复的信号
    private readonly ManualResetEventSlim _pauseRequest = new ManualResetEventSlim(false);
    // 读取线程确认已暂停的信号
    private readonly ManualResetEventSlim _pausedConfirmed = new ManualResetEventSlim(false);
    // 写入请求队列(线程安全)
    private readonly ConcurrentQueue<Action> _writeQueue = new ConcurrentQueue<Action>();
    // 停止标志
    private volatile bool _stopRequested = false;
    // 写入专用线程
    private Thread _writeThread;

    public CrossThreadReadWriteController()
    {
        // 启动写入专用线程
        _writeThread = new Thread(ProcessWriteQueue);
        _writeThread.Start();
    }

    // 读取线程的循环任务
    public void ReadLoop()
    {
        while (!_stopRequested)
        {
            // 检查是否需要暂停
            if (_pauseRequest.IsSet)
            {
                // 确认已暂停,并等待恢复信号
                _pausedConfirmed.Set();
                _pauseRequest.Wait();
                _pausedConfirmed.Reset();
            }

            // 模拟读取操作
            Console.WriteLine($"[Read] {DateTime.Now:HH:mm:ss.fff} - Reading data...");
            Thread.Sleep(1000); // 模拟耗时操作
        }
        Console.WriteLine("[Read] Thread stopped.");
    }

    // 处理写入队列的专用线程
    private void ProcessWriteQueue()
    {
        while (!_stopRequested || !_writeQueue.IsEmpty)
        {
            if (_writeQueue.TryDequeue(out var writeAction))
            {
                // 触发暂停读取线程
                RequestPause();

                // 执行写入操作
                writeAction.Invoke();

                // 恢复读取线程
                ResumeRead();
            }
            else
            {
                Thread.Sleep(50); // 队列为空时短暂休眠
            }
        }
        Console.WriteLine("[Write] Thread stopped.");
    }

    // 跨线程提交写入请求
    public void SubmitWriteCommand(Action writeAction)
    {
        _writeQueue.Enqueue(writeAction);
    }

    // 请求暂停读取线程(线程安全)
    private void RequestPause()
    {
        _pauseRequest.Set();
        _pausedConfirmed.Wait(); // 等待读取线程确认暂停
    }

    // 恢复读取线程(线程安全)
    private void ResumeRead()
    {
        _pauseRequest.Reset();
    }

    // 停止所有线程
    public void Stop()
    {
        _stopRequested = true;
        _pauseRequest.Set(); // 确保读取线程退出等待
        _writeThread.Join(); // 等待写入线程结束
    }
}

// 使用示例
public class Program
{
    public static void Main()
    {
        var controller = new CrossThreadReadWriteController();

        // 启动读取线程
        var readThread = new Thread(controller.ReadLoop);
        readThread.Start();

        // 模拟多个线程触发写入操作
        for (int i = 0; i < 3; i++)
        {
            var threadId = i;
            new Thread(() =>
            {
                controller.SubmitWriteCommand(() =>
                {
                    Console.WriteLine($"[Write-{threadId}] {DateTime.Now:HH:mm:ss.fff} - Writing data...");
                    Thread.Sleep(500); // 模拟耗时操作
                });
            }).Start();
        }

        Thread.Sleep(5000); // 等待所有写入完成
        controller.Stop();
        readThread.Join();
        Console.WriteLine("Main thread exited.");
    }
}

关键改进解析

1.跨线程写入请求的提交

通过 SubmitWriteCommand 方法,任何线程均可提交写入操作:

public void SubmitWriteCommand(Action writeAction)
{
    _writeQueue.Enqueue(writeAction); // 线程安全入队
}

2.专用写入线程处理队列

写入操作由独立线程顺序处理,避免多线程并发写入冲突:

private void ProcessWriteQueue()
{
    while (!_stopRequested || !_writeQueue.IsEmpty)
    {
        if (_writeQueue.TryDequeue(out var writeAction))
        {
            RequestPause();   // 暂停读取线程
            writeAction();    // 执行写入
            ResumeRead();     // 恢复读取线程
        }
    }
}

3.双重信号确保原子性

通过 RequestPause 和 ResumeRead 方法封装暂停与恢复逻辑:

private void RequestPause()
{
    _pauseRequest.Set();      // 发送暂停信号
    _pausedConfirmed.Wait();  // 阻塞等待读取线程确认暂停
}

4.线程安全停止机制

通过 _stopRequested 标志和队列检查确保安全退出:

public void Stop()
{
    _stopRequested = true;
    _writeThread.Join(); // 等待写入线程处理完队列
}

运行效果

[Read] 14:30:01.123 - Reading data...
[Read] 14:30:02.124 - Reading data...
[Write-0] 14:30:03.125 - Writing data...
[Read] 14:30:03.626 - Reading data...
[Write-1] 14:30:04.127 - Writing data...
[Read] 14:30:04.628 - Reading data...
[Write-2] 14:30:05.129 - Writing data...
Main thread exited.

方案优势

特性说明
多线程安全通过 ConcurrentQueue 和信号量,支持任意线程触发写入操作。
顺序执行写入操作由专用线程顺序处理,避免并发冲突。
无锁读取读取线程在非写入状态下完全无锁,最大化性能。
精准控制通过双信号机制确保写入操作执行期间读取线程完全暂停。

适用场景

  • 分布式任务调度:多个工作线程提交写入请求,由中心线程处理。
  • 实时数据采集:采集线程持续读取数据,外部线程动态更新配置。
  • 高并发服务:如网络服务器,处理来自不同客户端的并发更新操作。

注意事项

  • 队列积压风险:若写入操作频率过高,需监控队列长度或添加背压机制。
  • 异常处理:在写入操作中需捕获异常,避免导致写入线程崩溃。
  • 性能调优:可根据场景调整 Thread.Sleep 时间或使用无等待策略。

到此这篇关于C#中实现跨线程写入的示例代码的文章就介绍到这了,更多相关C# 跨线程写入内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C#连接到sql server2008数据库的实例代码

    C#连接到sql server2008数据库的实例代码

    这篇文章主要介绍了C#连接到sql server2008数据库的实例代码,需要的朋友可以参考下
    2017-09-09
  • C#实现读取被进程占用的文件实现方法

    C#实现读取被进程占用的文件实现方法

    这篇文章主要介绍了C#实现读取被进程占用的文件实现方法,涉及C#进程操作及文件读取的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-08-08
  • C#创建临时文件的方法

    C#创建临时文件的方法

    这篇文章主要介绍了C#创建临时文件的方法,涉及C#中Path.GetTempFileName方法获取与操作临时文件的使用技巧,非常具有实用价值,需要的朋友可以参考下
    2015-04-04
  • C#多线程与异步的区别详解

    C#多线程与异步的区别详解

    多线程和异步操作两者都可以达到避免调用线程阻塞的目的,从而提高软件的可响应性。甚至有些时候我们就认为多线程和异步操作是等同的概念。但是,多线程和异步操作还是有一些区别的。而这些区别造成了使用多线程和异步操作的时机的区别
    2017-06-06
  • C#通过HttpWebRequest发送带有JSON Body的POST请求实现

    C#通过HttpWebRequest发送带有JSON Body的POST请求实现

    本文主要介绍了C#通过HttpWebRequest发送带有JSON Body的POST请求实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09
  • Silverlight文件上传下载实现方法(下载保存)

    Silverlight文件上传下载实现方法(下载保存)

    这篇文章主要介绍了Silverlight文件上传下载实现方法(下载保存) ,需要的朋友可以参考下
    2015-11-11
  • P/Invoke之C#调用动态链接库DLL示例详解

    P/Invoke之C#调用动态链接库DLL示例详解

    这篇文章主要为大家介绍了P/Invoke之C#调用动态链接库DLL示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • C#自定义控件添加右键菜单的方法

    C#自定义控件添加右键菜单的方法

    这篇文章主要介绍了C#自定义控件添加右键菜单的方法,本文用到control控件,专门自定义右键菜单,下面小编给大家整理下,有需要的小伙伴可以来参考下
    2015-08-08
  • C#实现简单超市收银系统

    C#实现简单超市收银系统

    这篇文章主要为大家详细介绍了C#实现简单超市收银系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-02-02
  • C#实现奇偶排序的示例代码

    C#实现奇偶排序的示例代码

    奇偶排序是一种简单的排序算法,该算法通过比较并交换相邻的元素来完成排序,本文主要介绍了C#实现奇偶排序的示例代码,具有一定的参考价值,感兴趣的可以了解一下
    2023-11-11

最新评论