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

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.xdnf.cn/news/477145.html

相关文章:

  • 使用哈希表封装myunordered_set和myunordered_map
  • 【Python CGI编程】
  • go-中间件的使用
  • Spring MVC 对 JavaWeb 的优化:从核心组件到注解
  • CentOS系统中升级Python 3.12.2版本
  • 数据通信原理 光纤通信 期末速成
  • 【Umi】项目初始化配置和用户权限
  • Chrome拓展(Chrome Extension)开发定时任务插件
  • 掌握 Kotlin Android 单元测试:MockK 框架深度实践指南
  • npm cross-env工具包介绍(跨平台环境变量设置工具)
  • A2A vs MCP vs AG-UI
  • 在微创手术中使用Kinova轻型机械臂进行多视图图像采集和3D重建
  • CSS实现文本自动平衡text-wrap: balance
  • 容器化-k8s-使用和部署
  • 能源数字化转型关键引擎:Profinet转Modbus TCP网关驱动设备协同升级
  • 从XSS到“RCE“的PC端利用链构建
  • 消防应急处置管理的全流程概述
  • 软件架构风格系列(3):管道 - 过滤器架构
  • 顶层架构 - 消息集群推送方案
  • 【风控】用户特征画像体系
  • 第一次做逆向
  • oracle linux 95 升级openssh 10 和openssl 3.5 过程记录
  • kubernetes的service与服务发现
  • 7 个正则化算法完整总结
  • 快慢指针算法(Floyd 判圈算法)
  • pytorch 15.1 学习率调度基本概念与手动实现方法
  • SSH认证核心机制
  • 索恩格汽车SEG Automotive EDI 需求分析
  • 用 Rust 带你了解 TCP 和 UDP
  • Neo4j(一) - Neo4j安装教程(Windows)