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

cs 复制代码
/// <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文库

相关推荐
rockey6273 小时前
AScript如何实现中文脚本引擎
c#·.net·script·eval·expression·function·动态脚本
科技小花4 小时前
数据治理平台架构演进观察:AI原生设计如何重构企业数据管理范式
数据库·重构·架构·数据治理·ai-native·ai原生
一江寒逸4 小时前
零基础从入门到精通MySQL(中篇):进阶篇——吃透多表查询、事务核心与高级特性,搞定复杂业务SQL
数据库·sql·mysql
D4c-lovetrain4 小时前
linux个人心得22 (mysql)
数据库·mysql
阿里小阿希5 小时前
CentOS7 PostgreSQL 9.2 升级到 15 完整教程
数据库·postgresql
我是唐青枫5 小时前
C#.NET gRPC 深入解析:Proto 定义、流式调用与服务间通信取舍
开发语言·c#·.net
荒川之神5 小时前
Oracle 数据仓库雪花模型设计(完整实战方案)
数据库·数据仓库·oracle
做个文艺程序员5 小时前
MySQL安全加固十大硬核操作
数据库·mysql·安全
不吃香菜学java5 小时前
Redis简单应用
数据库·spring boot·tomcat·maven
unicrom_深圳市由你创科技6 小时前
做虚拟示波器这种实时波形显示的上位机,用什么语言?
c++·python·c#