SmartDapper.Repository

SmartDapper.Repository 是基于 SmartDapper 的 Repository + UnitOfWork 封装,提供一致的 CRUD / 分页 API,并统一事务边界(支持自动提交/回滚与嵌套事务复用),用于组织数据访问层代码。

功能特性

  • 泛型仓储IRepository<T> + Repository<T>(内置常用 CRUD/分页)
  • 工作单元IUnitOfWork + UnitOfWork(统一连接与事务)
  • 事务作用域ITransactionScope + TransactionScope(未 Complete() 自动回滚;支持复用外层事务)
  • 事务快捷封装ExecuteInTransaction* 扩展方法,一行包裹事务边界

安装

bash 复制代码
dotnet add package SmartDapper.Repository

快速开始(ASP.NET Core)

方式 A:只使用 Repository + UnitOfWork(不依赖中间件)

Program.cs 统一注册 IDbConnection / IUnitOfWork / IRepository<>(推荐):

csharp 复制代码
using System.Data;
using Microsoft.Data.SqlClient;
using SmartDapper.Repository.Extensions;

var builder = WebApplication.CreateBuilder(args);

var connectionString = builder.Configuration.GetConnectionString("Default")!;

builder.Services.AddSmartDapperRepository(
    connectionString,
    cs => new SqlConnection(cs));

提示:如果不是 SQL Server,请自行替换连接类型,或实现你自己的 connectionFactory

方式 A2(推荐):配合 AddSmartDapperConnections(多连接工厂)

如果你已经使用 SmartDapper 的多连接工厂(AddSmartDapperConnections),可以直接让 Repository 复用工厂配置:

csharp 复制代码
using Microsoft.Data.SqlClient;
using SmartDapper.Extensions;
using SmartDapper.Repository.Extensions;

var builder = WebApplication.CreateBuilder(args);

builder.Services.AddSmartDapperConnections(o =>
{
    o.Add("Main", builder.Configuration.GetConnectionString("Main")!, cs => new SqlConnection(cs));
    o.DefaultKey = "Main";
});

// 默认连接(CreateDefault)
builder.Services.AddSmartDapperRepository();

// 或指定 key(Create("Main"))
// builder.Services.AddSmartDapperRepository("Main");

方式 B:使用 SmartDapper.Middleware(包含 DI 注册 + 声明式事务)

如果你的项目是 Web API,并希望通过 [UnitOfWork] 自动开启事务,推荐直接用中间件包:

csharp 复制代码
using Microsoft.Data.SqlClient;
using SmartDapper.Middleware.Extensions;
using SmartDapper.Repository.Extensions;

var builder = WebApplication.CreateBuilder(args);
var connectionString = builder.Configuration.GetConnectionString("Default")!;

// 注册仓储 + 工作单元
builder.Services.AddSmartDapperRepository(connectionString, cs => new SqlConnection(cs));

var app = builder.Build();

// 注册工作单元中间件(可选:配合 [UnitOfWork])
app.UseSmartDapperUnitOfWork();

app.MapControllers();
app.Run();

基础用法

csharp 复制代码
public class CustomerService
{
    private readonly IRepository<Customer> _repo;

    public CustomerService(IRepository<Customer> repo)
    {
        _repo = repo;
    }

    public async Task<long> CreateAsync(Customer customer)
    {
        // 插入并返回自增 ID(如果你的实体/表是自增主键)
        return await _repo.InsertAndGetIdAsync(customer);
    }

    public Task<List<Customer>> ListAsync()
        => _repo.GetAllListAsync();
}

事务用法(推荐)

推荐 1:扩展方法 ExecuteInTransaction*(最简洁)

csharp 复制代码
using SmartDapper.Repository.Extensions;

await _unitOfWork.ExecuteInTransactionAsync(async () =>
{
    await _customerRepository.InsertAsync(customer);
    await _orderRepository.InsertAsync(order);
});

推荐 2:CreateTransactionScope(可控且支持嵌套复用)

csharp 复制代码
await using var scope = _unitOfWork.CreateTransactionScope();
try
{
    await _customerRepository.InsertAsync(customer);
    await _orderRepository.InsertAsync(order);

    await scope.CompleteAsync(); // 提交
}
catch
{
    // 未 Complete 时:DisposeAsync 自动回滚
    throw;
}

嵌套说明:如果外层已存在事务,TransactionScope 会自动复用外层事务,不会重复开启新事务。

不推荐:直接 BeginTransaction(需要你手动 Commit/Rollback)

如果你确实需要"完全手动控制提交/回滚",可以使用 BeginTransaction/Commit/Rollback(请务必显式提交/回滚,避免长事务)。

生命周期与资源释放(重要)

  • 典型 Web 场景IUnitOfWork / IDbConnection 建议按 Scoped 注册(每个请求一个实例)。
  • 请求结束释放 :DI 容器会调用 Dispose/DisposeAsync;若发现"未结束事务",UnitOfWork 会进行防御性回滚并释放连接资源。
  • 后台任务/非 HTTP 场景 :请手动创建 DI Scope,并在 scope 生命周期内使用同一个 IUnitOfWork
相关推荐
Murphy20238 分钟前
.net8 Swashbuckle.AspNetCore WEBAPI 配置要点记录
.net·swagger·webapi·swashbuckle
余衫马1 小时前
Agent Skills 实战(.NET):理论 × 代码 × 企业案例
人工智能·.net·agent·skill·openclaw
CSharp精选营3 小时前
字符串拼接用“+”还是 StringBuilder?别再凭感觉写了
.net·stringbuilder·字符串拼接·csharp
.NET修仙日记4 小时前
Acme .NET 工具类库:一站式解决.NET开发高频场景问题
.net·nuget·acme·.net8.0·.net9.0·acme.net·.net10.0
.NET修仙日记5 小时前
Acme.ReturnOh:让.NET API返回值处理更优雅,统一响应格式一步到位
c#·.net·webapi
喵叔哟6 小时前
19-AIAgent智能代理开发
微服务·.net
唐青枫9 小时前
深入理解 C#.NET TaskScheduler:为什么大量使用 Work-Stealing
c#·.net
喵叔哟9 小时前
20-多模态AI应用开发
人工智能·微服务·.net
桑榆肖物10 小时前
.NET 10 Native AOT 在 Linux 嵌入式设备上的实战
java·linux·.net·aot
我是唐青枫10 小时前
深入理解 C#.NET Task.Run:调度原理、线程池机制与性能优化
性能优化·c#·.net