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

SqlHelper 实现类,支持多数据库,提供异步操作、自动重试、事务、存储过程、分页、缓存等功能。

/// <summary>
/// SqlHelper 实现类,支持多数据库,提供异步操作、自动重试、事务、存储过程、分页、缓存等功能。
/// </summary>
public class SqlHelper : IDbHelper
{private readonly IDbConnectionFactory _connectionFactory;private readonly ILogger<SqlHelper>? _logger;private readonly AsyncPolicy _retryPolicy;private readonly IRedisCacheHelper? _redisCacheHelper;/// <summary>/// 构造函数,注入数据库连接工厂,日志记录器,自动重试次数与延迟,以及可选 Redis 缓存。/// </summary>/// <param name="connectionFactory">数据库连接工厂</param>/// <param name="logger">日志记录器</param>///<param name="retryCount">自动重试次数</param> /// <param name="retryDelayMs">重试延迟(毫秒)</param> /// <param name="redisCacheHelper">Redis 缓存(可空)</param> public SqlHelper( IDbConnectionFactory connectionFactory, ILogger<SqlHelper>? logger = null, int retryCount = 3, int retryDelayMs = 200, IRedisCacheHelper? redisCacheHelper = null) { _connectionFactory = connectionFactory; _logger = logger;_redisCacheHelper = redisCacheHelper;_retryPolicy = Policy.Handle<Exception>().WaitAndRetryAsync(retryCount, _ => TimeSpan.FromMilliseconds(retryDelayMs),(ex, timespan, retryCount, context) =>{_logger?.LogWarning($"Retry {retryCount} in {timespan.TotalMilliseconds}ms due to: {ex.Message}");});}/// <summary>/// 获取数据库连接实例/// </summary>/// <returns>DbConnection 实例</returns>private DbConnection GetConnection() => _connectionFactory.CreateConnection();public async Task<IEnumerable<T>> QueryAsync<T>(string sql, object? parameters = null){var sw = Stopwatch.StartNew();await using var conn = GetConnection();await conn.OpenAsync();var result = await _retryPolicy.ExecuteAsync(() => conn.QueryAsync<T>(sql, parameters));sw.Stop();_logger?.LogInformation($"[Query] {sw.ElapsedMilliseconds}ms - SQL: {sql}");return result;}public async Task<T> QueryFirstOrDefaultAsync<T>(string sql, object? parameters = null){var sw = Stopwatch.StartNew();await using var conn = GetConnection();await conn.OpenAsync();var result = await _retryPolicy.ExecuteAsync(() => conn.QueryFirstOrDefaultAsync<T>(sql, parameters));sw.Stop();_logger?.LogInformation($"[QueryFirstOrDefault] {sw.ElapsedMilliseconds}ms - SQL: {sql}");return result;}public async Task<int> ExecuteAsync(string sql, object? parameters = null){var sw = Stopwatch.StartNew();await using var conn = GetConnection();await conn.OpenAsync();var result = await _retryPolicy.ExecuteAsync(() => conn.ExecuteAsync(sql, parameters));sw.Stop();_logger?.LogInformation($"[Execute] {sw.ElapsedMilliseconds}ms - SQL: {sql}");return result;}public async Task<IEnumerable<T>> RunProcedureAsync<T>(string procName, object? parameters = null){var sw = Stopwatch.StartNew();await using var conn = GetConnection();await conn.OpenAsync();var result = await _retryPolicy.ExecuteAsync(() => conn.QueryAsync<T>(procName, parameters, commandType: CommandType.StoredProcedure));sw.Stop();_logger?.LogInformation($"[RunProcedure] {sw.ElapsedMilliseconds}ms - Procedure: {procName}");return result;}public async Task ExecuteInTransactionAsync(Func<IDbHelper, Task> transactionalOperations){await using var conn = GetConnection();await conn.OpenAsync();await using var tran = await conn.BeginTransactionAsync();try{var tranHelper = new SqlHelperWithTransaction(conn, tran, _connectionFactory.DbProvider, _logger, _retryPolicy, _redisCacheHelper);await transactionalOperations(tranHelper);await tran.CommitAsync();}catch (Exception ex){await tran.RollbackAsync();_logger?.LogError(ex, "Transaction failed.");throw;}}public Task<string> BuildPagedQuery(string baseSql, string orderBy, int pageIndex, int pageSize){int offset = (pageIndex - 1) * pageSize;string pagedSql = _connectionFactory.DbProvider.ToLower() switch{"mysql" => $"{baseSql} ORDER BY {orderBy} LIMIT {offset}, {pageSize}","postgresql" => $"{baseSql} ORDER BY {orderBy} OFFSET {offset} LIMIT {pageSize}",_ => $"{baseSql} ORDER BY {orderBy} OFFSET {offset} ROWS FETCH NEXT {pageSize} ROWS ONLY"};return Task.FromResult(pagedSql);}public async Task<IEnumerable<T>> QueryCachedAsync<T>(string cacheKey, string sql, object? parameters = null, TimeSpan? expiry = null){if (_redisCacheHelper == null)throw new InvalidOperationException("Redis cache helper not configured.");return await _redisCacheHelper.GetOrSetAsync(cacheKey, async () =>{await using var conn = GetConnection();await conn.OpenAsync();return await conn.QueryAsync<T>(sql, parameters);}, expiry);}}

完整源码:【免费】.Net7封装操作数据库,支持MySql、MSSQL、postgresql多数据库操作资源-CSDN文库

http://www.lryc.cn/news/2380242.html

相关文章:

  • 语音识别——声纹识别
  • window 显示驱动开发-报告图形内存(三)
  • 安全运维 -- linux磁盘挂载到windows
  • 使用 Apache POI 生成 Word 文档
  • 高防服务器流量“清洗”什么意思
  • UE5 GAS框架解析内部数据处理机制——服务器与客户端
  • Unity实用技能-UI定位总结
  • 开源GPU架构RISC-V VCIX的深度学习潜力测试:从RTL仿真到MNIST实战
  • 服务间的“握手”:OpenFeign声明式调用与客户端负载均衡
  • 26、DAPO论文笔记(解耦剪辑与动态采样策略优化,GRPO的改进)
  • JQuery 禁止页面滚动(防止页面抖动)
  • Android Coli 3 ImageView load two suit Bitmap thumb and formal,Kotlin(七)
  • Halcon与C#:工业级机器视觉开发
  • Unity序列化字段、单例模式(Singleton Pattern)
  • 【工具】Windows|外接的显示器怎么用软件调亮度(Brightness Slider)
  • 在 Java MyBatis 中遇到 “操作数类型冲突: varbinary 与 float 不兼容” 的解决方法
  • 系统架构设计(十四):解释器风格
  • 【Nextcloud】使用 LNMP 架构搭建私有云存储:Nextcloud 实战指南
  • VDC、SMC、MCU怎么协同工作的?
  • 【办公类-100-01】20250515手机导出教学照片,自动上传csdn+最小化Vscode界面
  • Java-List集合类全面解析
  • uniapp-商城-60-后台 新增商品(属性的选中和页面显示,数组join 的使用)
  • [c语言日寄]数据结构:栈
  • WEB安全--Java安全--LazyMap_CC1利用链
  • 【杂谈】-AI 重塑体育营销:从内容管理到创意释放的全面变革
  • 黑马k8s(六)
  • 【数据结构】二分查找(返回插入点)5.14
  • 如何设计一个二级缓存(Redis+Caffeine)架构?Redis 6.0多线程模型如何工作?
  • Java:logback-classic与slf4j版本对应关系
  • 【OpenGL学习】(一)创建窗口