C#开发基础之单例模式下的集合数据,解决并发访问读写冲突的问题

news2024/9/22 10:05:19

在这里插入图片描述

1. 前言

在C#中,使用单例模式管理集合数据时,如果多线程同时访问集合,容易产生并发访问的读写冲突问题。单例模式下集合数据的并发访问读写冲突是如何产生的?

单例模式确保一个类在整个应用运行期间只有一个实例,这使得单例对象成为全局共享的资源。当这个单例对象包含集合数据(如List、Dictionary等),并且这些集合数据被多个线程同时访问时,就可能产生并发读写冲突。

2. 并发读写冲突

并发读写冲突通常发生在以下几种情况:

  • 写-写冲突:两个或多个线程同时尝试修改集合中的数据。例如,一个线程正在向集合中添加元素,而另一个线程试图删除元素,或者两个线程同时添加元素,可能导致元素丢失、重复或破坏集合的内部结构。
  • 读-写冲突:一个线程正在读取集合中的数据,而另一个线程同时修改集合(如添加、删除或更新元素)。这可能导致读取线程看到不一致的数据状态,即所谓的“脏读”。
  • 复合操作的不完整性:当一个操作需要多个步骤完成(如先检查元素是否存在再删除),如果在这过程中被其他线程打断,可能导致不符合预期的结果。
    在单例模式下,如果没有适当的并发控制措施(如锁、同步块、线程安全集合等),这些问题就会变得尤为突出,因为所有对集合的操作都集中在一个共享实例上。

3. 解决策略

为了避免这些问题,可以采取以下几种策略:

  • 使用锁:在访问集合的读写操作前后加锁,确保同一时间只有一个线程能访问集合。这可以是传统的锁机制,如C#中的lock语句关键字。
  • 使用线程安全的集合:许多编程语言提供了专为并发访问设计的线程安全集合,如C#中的ConcurrentDictionary等,这些集合内部实现了必要的同步机制。
  • 不可变集合:使用不可变集合可以在一定程度上避免写冲突,因为一旦创建,集合就不能被改变,只能通过操作返回新的集合实例。但这通常适用于读多写少的场景。
  • 副本和快照:在需要修改集合时,先创建集合的一个副本,然后在副本上操作,最后替换原集合。这种方式减少了锁的范围和时间,但可能增加内存开销。

4. 编码实践

1. 使用lock关键字

在这里插入图片描述
可以使用 lock 关键字来同步对集合的访问。lock 能确保在同一时刻只有一个线程能够访问被锁定的代码块,从而避免并发冲突。

public sealed class Singleton
{
    private static readonly Singleton instance = new Singleton();
    private List<int> data = new List<int>();
    private static readonly object lockObject = new object();

    private Singleton() {}

    public static Singleton Instance
    {
        get { return instance; }
    }

    public void AddData(int value)
    {
        lock (lockObject)
        {
            data.Add(value);
        }
    }

    public List<int> GetData()
    {
        lock (lockObject)
        {
            // 返回副本,防止外部修改原始数据
            return new List<int>(data);
        }
    }
}

2. 使用ConcurrentDictionary或ConcurrentBag

在这里插入图片描述
.NET 提供了一些线程安全的集合类,如 ConcurrentDictionary 和 ConcurrentBag。这些集合类内置了线程同步机制,可以直接使用。

public sealed class Singleton
{
    private static readonly Singleton instance = new Singleton();
    private ConcurrentBag<int> data = new ConcurrentBag<int>();

    private Singleton() {}

    public static Singleton Instance
    {
        get { return instance; }
    }

    public void AddData(int value)
    {
        data.Add(value);
    }

    public List<int> GetData()
    {
        return data.ToList();
    }
}

3. 使用ReaderWriterLockSlim

在这里插入图片描述

ReaderWriterLockSlim 允许多个线程同时读取,但写操作是独占的。这种锁非常适合读多写少的场景。

public sealed class Singleton
{
    private static readonly Singleton instance = new Singleton();
    private List<int> data = new List<int>();
    private static readonly ReaderWriterLockSlim lockSlim = new ReaderWriterLockSlim();

    private Singleton() {}

    public static Singleton Instance
    {
        get { return instance; }
    }

    public void AddData(int value)
    {
        lockSlim.EnterWriteLock();
        try
        {
            data.Add(value);
        }
        finally
        {
            lockSlim.ExitWriteLock();
        }
    }

    public List<int> GetData()
    {
        lockSlim.EnterReadLock();
        try
        {
            return new List<int>(data);
        }
        finally
        {
            lockSlim.ExitReadLock();
        }
    }
}

4. 使用ImmutableCollection

在这里插入图片描述

ImmutableCollection 提供了线程安全的集合操作。集合的每次修改都会创建一个新的集合,原有集合保持不变。

public sealed class Singleton
{
    private static readonly Singleton instance = new Singleton();
    private ImmutableList<int> data = ImmutableList<int>.Empty;
    private static readonly object lockObject = new object();

    private Singleton() {}

    public static Singleton Instance
    {
        get { return instance; }
    }

    public void AddData(int value)
    {
        lock (lockObject)
        {
            data = data.Add(value);
        }
    }

    public ImmutableList<int> GetData()
    {
        // 不需要锁,因为 ImmutableList 是不可变的
        return data;
    }
}

5. 总结

以上几种方法可以有效解决单例模式下集合数据的并发访问问题。具体选择哪种方法取决于你的应用场景和性能需求。如果写操作较少,读操作较多,可以选择 ReaderWriterLockSlim 或 ImmutableCollection。如果需要较高的写性能,可以考虑使用 ConcurrentDictionary 或 ConcurrentBag。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2139358.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CSP-J 算法基础 图论

文章目录 前言图的简介1. **图的定义**2. **图的类型**3. **图的表示方法**a. **邻接矩阵&#xff08;Adjacency Matrix&#xff09;**b. **邻接表&#xff08;Adjacency List&#xff09;** 4. **图的基本操作**5. **图的遍历**6. **图的应用**7. **图的算法** 出度与入度1. *…

Android实现关机和重启功能

文章目录 需求场景需求场景经历 一、解决思路和方案实际困难点情况普遍思路用Shell 命令实现应用和系统联调遇到的问题 个人解决思路和方案 二、代码跟踪系统实现的关机、重启界面GlobalActionsDialogLite.java 创建关机、重启菜单createActionItems()shutdownAction GlobalAct…

时序数据库 TDengine 的入门体验和操作记录

时序数据库 TDengine 的学习和使用经验 什么是 TDengine &#xff1f;什么是时序数据 &#xff1f;使用RPM安装包部署默认的网络端口 TDengine 使用TDengine 命令行&#xff08;CLI&#xff09;taosBenchmark服务器内存需求删库跑路测试 使用体验文档纠错 什么是 TDengine &…

GEE:连续变化检测与分类(Continuous Change Detection and Classification, CCDC)教程

连续变化检测与分类&#xff08;Continuous Change Detection and Classification, CCDC&#xff09;是一种土地变化监测算法&#xff0c;旨在对卫星数据的时间序列进行操作&#xff0c;特别是Landsat数据。CCDC包括两个部分&#xff0c;其一是变化检测算法&#xff08;Change …

Mybatis中Like模糊查询三种处理方式

目录 Mybatis中Like模糊查询三种处理方式 1.通过单引号拼接${} 1&#xff09;mapper接口 2&#xff09;Mapper.xml 3&#xff09;测试代码 4) 测试结果 2.通过concat()函数拼接(个人推荐使用这种) 1&#xff09;mapper接口 2&#xff09;Mapper.xml 3&#xff09;测试代码 4) 测…

C语言-整数和浮点数在内存中的存储-详解-下

C语言-整数和浮点数在内存中的存储-详解-下 1.前言2.浮点数2.1IEEE 754 标准2.2存储格式存储细节取出 3.相关代码的解释 1.前言 在C语言-整数和浮点数在内存中的存储-详解-上中&#xff0c;我通过一个简单的例子展示了整数和浮点数在内存中的存储差异&#xff0c;并详细介绍了…

Java重修笔记 第五十六天 坦克大战(六)多线程基础 - 线程同步、死锁

多线程同步机制 多线程编程中&#xff0c;一些敏感数据可能会被多个线程同时访问造成数据混乱&#xff08;例如票数&#xff09;&#xff0c;使用线程同步机制&#xff0c;通过锁对象&#xff08;对象实例或类实例&#xff09;的方式来保证该段代码在任意时刻&#xff0c;最多…

TCP socket

TCP的socket和UDP大同小异&#xff0c;基本的代码结构都是相同的。一些相同的接口本文就不赘述了&#xff0c;例如&#xff0c;socket,bind&#xff0c;有需要看这篇文章UDP socket 服务端server 两步&#xff1a;初始化服务端&#xff0c;运行服务端 初始化服务端 创建soc…

Java项目基于docker 部署配置

linux新建文件夹 data cd datatouch Dockerfilesudo vim Dockerfile# 使用一个基础的 Java 镜像&#xff08;根据自己项目中使用的是什么jdk版本设置&#xff0c;用于拉取执行jar包的jdk环境&#xff09; FROM openjdk:8# 指定工作目录 VOLUME /data# 复制应用程序的 JAR 文件…

Redis模拟消息队列实现异步秒杀

目录 一、消息队列含义 二、Redis实现消息队列 1、基于List的结构模拟实现消息队列 2、基于PubSub的消息队列 3、基于Stream的消息队列 4、基于Stream的消息队列- 消费者组 一、消息队列含义 消息队列&#xff08;Message Queue&#xff09;&#xff0c;字面意思就是存放…

基于SpringBoot的招生宣传管理系统【附源码】

基于SpringBoot的招生宣传管理系统&#xff08;源码L文说明文档&#xff09; 目录 4 系统设计 4.1 系统概述 4.2系统功能结构设计 4.3数据库设计 4.3.1数据库E-R图设计 4.3.2 数据库表结构设计 5 系统实现 5.1管理员功能介绍 5.1.1管理员登录 …

【JavaScript】LeetCode:36-40

文章目录 36 两数相加37 删除链表的倒数第n个节点38 两两交换链表中的节点39 k个一组翻转链表40 随机链表的复制 36 两数相加 创建一个新的链表&#xff08;哨兵节点指向&#xff09;&#xff0c;这个链表用来表示两个数相加后的和。从个位开始相加&#xff0c;每次都向新链表尾…

PCL 点云随机渲染颜色

目录 一、概述 1.1原理 1.2实现步骤 1.3 应用场景 二、代码实现 2.1关键函数 2.2完整代码 三、实现效果 PCL点云算法汇总及实战案例汇总的目录地址链接&#xff1a; PCL点云算法与项目实战案例汇总&#xff08;长期更新&#xff09; 一、概述 本文将介绍如何使用PCL库…

PMP–一、二、三模–分类–14.敏捷–技巧–项目生命周期

文章目录 技巧项目生命周期 一模14.敏捷--项目生命周期--原型法--迭代型生命周期&#xff0c;通过连续的原型或概念验证来改进产品或成果。每个新的原型都能带来新的干系人新的反馈和团队见解。题目中明确提到需要反馈&#xff0c;因此原型法比较好用。23、 [单选] 一个敏捷团队…

DAY13信息打点-Web 应用源码泄漏开源闭源指纹识别GITSVNDS备份

#知识点 0、Web架构资产-平台指纹识别 1、开源-CMS指纹识别源码获取方式 2、闭源-习惯&配置&特性等获取方式 3、闭源-托管资产平台资源搜索监控 演示案例&#xff1a; ➢后端-开源-指纹识别-源码下载 ➢后端-闭源-配置不当-源码泄漏 ➢后端-方向-资源码云-源码泄漏 …

Linux下的简单TCP客户端和服务器

客户端 #include <arpa/inet.h> #include <stdlib.h> #include <stdio.h> #include <string.h> #include <unistd.h> #include <sys/socket.h>int main() {struct sockaddr_in* caddr(struct sockaddr_in*)malloc(sizeof(struct sockaddr…

【JavaEE】IO基础知识及代码演示

目录 一、File 1.1 观察get系列特点差异 1.2 创建文件 1.3.1 delete()删除文件 1.3.2 deleteOnExit()删除文件 1.4 mkdir 与 mkdirs的区别 1.5 文件重命名 二、文件内容的读写----数据流 1.1 InputStream 1.1.1 使用 read() 读取文件 1.2 OutputStream 1.3 代码演示…

【有啥问啥】自动提示词工程(Automatic Prompt Engineering, APE):深入解析与技术应用

自动提示词工程&#xff08;Automatic Prompt Engineering, APE&#xff09;&#xff1a;深入解析与技术应用 引言 随着大语言模型&#xff08;LLM&#xff09;如 GPT、BERT 等的快速发展&#xff0c;如何高效地与这些模型进行互动成为了重要的研究方向之一。提示词&#xff…

阿里P8和P9级别有何要求

阿里巴巴的P8和P9级别&#xff0c;代表着公司的资深技术专家或管理者岗位&#xff0c;要求候选人具有丰富的职业经历、深厚的技术能力以及出色的领导力。以下是对P8和P9级别的要求、考察点以及准备建议的详细分析。 P8 级别要求 1. 职业经历&#xff1a; 8年以上的工作经验&a…

PCIe进阶之TL:Common Packet Header Fields TLPs with Data Payloads Rules

1 Transaction Layer Protocol - Packet Definition TLP有四种事务类型:Memory、I/O、Configuration 和 Messages,两种地址格式:32bit 和 64bit。 构成 TLP 时,所有标记为 Reserved 的字段(有时缩写为 R)都必须全为0。接收者Rx必须忽略此字段中的值,PCIe Switch 必须对…