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

c# 深度解析:实现一个通用配置管理功能,打造高并发、可扩展的配置管理神器

文章目录

  • 深入分析 ConfigManager<TKey, TValue> 类
    • 1. 类设计概述
    • 2. 核心成员分析
      • 2.1 字段和属性
      • 2.2 构造函数
    • 3. 数据加载机制
    • 4. CRUD 操作方法
      • 4.1 添加数据
      • 4.2 删除数据
      • 4.3 更新数据
      • 4.4 查询数据
      • 4.5 清空数据
    • 5. 数据持久化
    • 6. 设计亮点
    • 7. 使用示例

ConfigManager<TKey, TValue> 是一个泛型抽象类,用于管理键值对配置数据的加载、存储和操作。下面将从多个维度全面分析这个类的设计和实现。

public abstract class ConfigManager<TKey, TValue>
{private readonly ConcurrentDictionary<TKey, TValue> _datas;protected abstract string ModulePath { get; }private readonly string _path;public IReadOnlyDictionary<TKey, TValue> Datas => _datas;protected ConfigManager(){_path = $"{path1}/{path2}{ModulePath}";_datas = LoadData() ?? new ConcurrentDictionary<TKey, TValue>();}private ConcurrentDictionary<TKey, TValue> LoadData(){try{var loadedData = JsonHelper_S.GetData<Dictionary<TKey, TValue>>(_path);return loadedData != null? new ConcurrentDictionary<TKey, TValue>(loadedData): new ConcurrentDictionary<TKey, TValue>();}catch (Exception ex){Console.WriteLine($"Error loading config data: {ex.Message}");return new ConcurrentDictionary<TKey, TValue>();}}public void Add(TKey key, TValue value){if (_datas.TryAdd(key, value)){//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));}}public bool Remove(TKey key){bool removed = _datas.TryRemove(key, out _);if (removed){//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));}return removed;}public bool TryGetValue(TKey key, out TValue value){return _datas.TryGetValue(key, out value);}public void Update(TKey key, TValue value){_datas.AddOrUpdate(key, value, (k, oldValue) => value);//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));}public void Clear(){_datas.Clear();//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));}public void Save(){Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));}
}

深入分析 ConfigManager<TKey, TValue> 类

ConfigManager<TKey, TValue> 是一个泛型抽象类,用于管理键值对配置数据的加载、存储和操作。下面我将从多个维度全面分析这个类的设计和实现。

1. 类设计概述

这是一个抽象基类,使用泛型参数 TKeyTValue 来支持不同类型的键值对配置。主要特点包括:

  • 使用 ConcurrentDictionary 保证线程安全
  • 抽象路径定义,由子类实现具体配置路径
  • 提供基本的 CRUD 操作
  • 自动持久化到 JSON 文件
  • 错误处理和日志记录

2. 核心成员分析

2.1 字段和属性

private readonly ConcurrentDictionary<TKey, TValue> _datas;
protected abstract string ModulePath { get; }
private readonly string _path;
public IReadOnlyDictionary<TKey, TValue> Datas => _datas;
  • _datas: 使用 ConcurrentDictionary 存储配置数据,确保线程安全
  • ModulePath: 抽象属性,子类必须实现,定义配置文件的相对路径
  • _path: 完整的配置文件路径,由构造函数初始化
  • Datas: 只读属性,提供对配置数据的只读访问

2.2 构造函数

protected ConfigManager()
{_path = $"{path1}/{path2}{ModulePath}";_datas = LoadData() ?? new ConcurrentDictionary<TKey, TValue>();
}
  • 构造完整路径:组合基础路径和模块路径
  • 初始化数据:调用 LoadData() 方法加载现有配置,失败则创建空字典

3. 数据加载机制

private ConcurrentDictionary<TKey, TValue> LoadData()
{try{var loadedData = JsonHelper_S.GetData<Dictionary<TKey, TValue>>(_path);return loadedData != null? new ConcurrentDictionary<TKey, TValue>(loadedData): new ConcurrentDictionary<TKey, TValue>();}catch (Exception ex){Console.WriteLine($"Error loading config data: {ex.Message}");return new ConcurrentDictionary<TKey, TValue>();}
}
  • 使用 JsonHelper_S.GetData 从 JSON 文件加载数据
  • 错误处理:捕获异常并记录错误日志
  • 回退机制:加载失败时返回空字典

4. CRUD 操作方法

4.1 添加数据

public void Add(TKey key, TValue value)
{if (_datas.TryAdd(key, value)){//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));}
}
  • 使用 TryAdd 确保线程安全
  • 添加成功后注释了异步保存逻辑(可能是为了性能考虑)

4.2 删除数据

public bool Remove(TKey key)
{bool removed = _datas.TryRemove(key, out _);if (removed){//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));}return removed;
}
  • 使用 TryRemove 安全删除
  • 返回操作结果
  • 删除成功后注释了异步保存

4.3 更新数据

public void Update(TKey key, TValue value)
{_datas.AddOrUpdate(key, value, (k, oldValue) => value);//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));
}
  • 使用 AddOrUpdate 原子操作
  • 确保更新操作的线程安全性
  • 注释了异步保存

4.4 查询数据

public bool TryGetValue(TKey key, out TValue value)
{return _datas.TryGetValue(key, out value);
}
  • 标准字典查询方法
  • 使用 TryGetValue 模式避免异常

4.5 清空数据

public void Clear()
{_datas.Clear();//Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));
}
  • 清空所有配置项
  • 注释了异步保存

5. 数据持久化

public void Save()
{Task.Run(() => JsonHelper_S.SaveData(_datas.ToDictionary(kvp => kvp.Key, kvp => kvp.Value), _path));
}
  • 显式保存方法
  • 使用 Task.Run 异步执行保存操作
  • ConcurrentDictionary 转换为普通 Dictionary 后保存

6. 设计亮点

  1. 线程安全:全程使用 ConcurrentDictionary 确保多线程环境下的安全性
  2. 抽象设计:通过抽象 ModulePath 属性支持不同模块的配置管理
  3. 错误处理:加载数据时有完善的错误处理和回退机制
  4. 惰性持久化:注释了自动保存,可能是为了避免频繁IO操作
  5. 只读接口:通过 IReadOnlyDictionary 暴露数据,防止外部修改

7. 使用示例

子类实现可能如下:

public class UserSettingsManager : ConfigManager<string, string>
{protected override string ModulePath => "/UserSettingsBase/UserSettingsConfigManagerIsLog";private static readonly Lazy<UserSettingsManager > _instance =new Lazy<UserSettingsManager >(() => new UserSettingsManager ());public static UserSettingsManager Instance => _instance.Value;private UserSettingsManager () : base() { }
}

使用示例:

var manager = new UserSettingsManager();
manager.Add("theme", "dark");
manager.Update("language", "en-US");
if(manager.TryGetValue("theme", out var theme))
{Console.WriteLine($"Current theme: {theme}");
}
manager.Save();
http://www.dtcms.com/a/278991.html

相关文章:

  • Typecho加密文章HTML结构自定义完全指南
  • 在 Windows 主机和 VMware 虚拟机中的 Ubuntu 系统之间实现复制粘贴,
  • Spring IoC 容器实战:从解耦到集成的 6 大核心应用场景
  • 【PTA数据结构 | C语言版】字符串插入操作(不限长)
  • 微前端框架深度对决:qiankun、micro-app、wujie 技术内幕与架构选型指南
  • Ubuntu-25.04 Wayland桌面环境安装Anaconda3之后无法启动anaconda-navigator问题解决
  • 如何降低AIGC的查重率?精选六个AIGC降重让论文更出色
  • Spring Boot项目结构解析:构建高效、清晰的代码框架
  • 【Python进阶】深度复制——deepcopy
  • STM32F1_Hal库学习EXTI
  • 苍穹外卖学习指南(java的一个项目)(老师能运行,但你不行,看这里!!)
  • 最近要上Android 15的高通平台,按照之前Android14的高通平台的裁剪APP的方法修改,发现一改编译之后就不能进系统
  • LLaMA.cpp HTTP 服务参数: --pooling 嵌入模型 池化类型详解
  • 笔试——Day7
  • Datawhale AI夏令营大模型 task2.1
  • QML 常用控件(二)
  • Qt小组件 - 3 imageLabel
  • 【CV综合实战】基于深度学习的工业压力表智能检测与读数系统【3】使用OpenCV读取分割后的压力表读数
  • 《C++内存泄漏8大战场:Qt/MFC实战详解 + 面试高频陷阱破解》
  • 机器学习中的朴素贝叶斯(Naive Bayes)模型
  • AI日报 - 2025年07月14日
  • 认识下计算机视觉中的人脸识别
  • 网络准入控制系统的作用解析,2025年保障企业入网安全第一道防线
  • 【邀请函】网易灵动露天矿山具身智能技术发布会,7月26日上海见
  • 【笔记】chrome 无法打开特定协议或访问特定协议时卡死
  • AI香烟检测实战:YOLO11模型训练全过程解析
  • 多尺度频率辅助类 Mamba 线性注意力模块(MFM),融合频域和空域特征,提升多尺度、复杂场景下的目标检测能力
  • Docker 拉取镜像并离线迁移至云桌面指南(以Redis为例)
  • 【API测试】Apifox、Apipost、Postman测试工具详解,Mock介绍
  • docker私有仓库