Dapper单元测试怎么写 Dapper仓储层单元测试方法

来源:草根站长作者:柬埔寨程序员头衔:程序员
导读:本期聚焦于小伙伴创作的《Dapper单元测试怎么写 Dapper仓储层单元测试方法》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《Dapper单元测试怎么写 Dapper仓储层单元测试方法》有用,将其分享出去将是对创作者最好的鼓励。

Dapper是一款轻量级的ORM框架,在.NET项目中常被用于仓储层的数据访问操作。对Dapper仓储层编写单元测试时,核心目标是验证仓储层的数据操作逻辑是否符合预期,同时避免依赖真实的数据库环境,减少测试的不确定性和执行成本。

Dapper单元测试怎么写 Dapper仓储层单元测试方法

Dapper仓储层单元测试的核心思路

仓储层的核心逻辑是调用Dapper的扩展方法执行SQL语句,完成数据的增删改查操作。要编写单元测试,首先需要解决两个问题:一是隔离真实的数据库连接,避免测试对生产数据库造成影响;二是模拟Dapper的方法调用,验证传入的SQL和参数是否正确,或者模拟返回预期的查询结果。

常见的实现方式有两种:一种是使用内存数据库(如SQLite的内存模式)替代真实数据库,执行真实的SQL逻辑;另一种是通过抽象数据库连接和Dapper的调用,使用Mock框架模拟相关依赖。

基于内存数据库的测试方案

SQLite的内存模式可以模拟真实的数据库环境,且数据仅在连接生命周期内存在,非常适合单元测试场景。首先需要定义仓储层的接口和基础实现:

// 仓储层接口定义
public interface IUserRepository
{
    Task<User> GetByIdAsync(int id);
    Task<int> AddAsync(User user);
}

// 用户实体类
public class User
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int Age { get; set; }
}

// 仓储层实现
public class UserRepository : IUserRepository
{
    private readonly IDbConnection _connection;

    public UserRepository(IDbConnection connection)
    {
        _connection = connection;
    }

    public async Task<User> GetByIdAsync(int id)
    {
        var sql = "SELECT Id, Name, Age FROM User WHERE Id = @Id";
        return await _connection.QueryFirstOrDefaultAsync<User>(sql, new { Id = id });
    }

    public async Task<int> AddAsync(User user)
    {
        var sql = "INSERT INTO User (Name, Age) VALUES (@Name, @Age); SELECT last_insert_rowid()";
        return await _connection.ExecuteScalarAsync<int>(sql, user);
    }
}

接下来编写单元测试,使用SQLite内存数据库作为数据源:

using Dapper;
using Microsoft.Data.Sqlite;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using System.Data;
using System.Threading.Tasks;

[TestClass]
public class UserRepositoryTests
{
    // 创建内存数据库连接
    private IDbConnection CreateInMemoryConnection()
    {
        var connection = new SqliteConnection("Data Source=:memory:");
        connection.Open();
        // 创建测试表
        connection.Execute(@"CREATE TABLE User (
            Id INTEGER PRIMARY KEY AUTOINCREMENT,
            Name TEXT NOT NULL,
            Age INTEGER NOT NULL
        )");
        return connection;
    }

    [TestMethod]
    public async Task GetByIdAsync_WhenUserExists_ReturnsUser()
    {
        // 准备
        using var connection = CreateInMemoryConnection();
        var repository = new UserRepository(connection);
        // 插入测试数据
        await connection.ExecuteAsync("INSERT INTO User (Name, Age) VALUES ('张三', 25)");

        // 执行
        var result = await repository.GetByIdAsync(1);

        // 断言
        Assert.IsNotNull(result);
        Assert.AreEqual("张三", result.Name);
        Assert.AreEqual(25, result.Age);
    }

    [TestMethod]
    public async Task AddAsync_ValidUser_ReturnsNewId()
    {
        // 准备
        using var connection = CreateInMemoryConnection();
        var repository = new UserRepository(connection);
        var user = new User { Name = "李四", Age = 30 };

        // 执行
        var newId = await repository.AddAsync(user);

        // 断言
        Assert.IsTrue(newId > 0);
        var insertedUser = await connection.QueryFirstOrDefaultAsync<User>("SELECT Id, Name, Age FROM User WHERE Id = @Id", new { Id = newId });
        Assert.IsNotNull(insertedUser);
        Assert.AreEqual("李四", insertedUser.Name);
    }
}

基于Mock框架的测试方案

如果不想依赖真实的数据库执行环境,可以使用Moq等Mock框架模拟IDbConnection和Dapper的方法调用,直接验证仓储层的逻辑是否符合预期:

using Dapper;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using Moq;
using System.Data;
using System.Threading.Tasks;

[TestClass]
public class UserRepositoryMockTests
{
    private Mock<IDbConnection> _mockConnection;
    private IUserRepository _repository;

    [TestInitialize]
    public void Setup()
    {
        _mockConnection = new Mock<IDbConnection>();
        _repository = new UserRepository(_mockConnection.Object);
    }

    [TestMethod]
    public async Task GetByIdAsync_CallsCorrectSqlAndParameters()
    {
        // 准备
        User expectedUser = null;
        _mockConnection.Setup(c => c.QueryFirstOrDefaultAsync<User>(
            It.IsAny<string>(),
            It.IsAny<object>(),
            It.IsAny<IDbTransaction>(),
            It.IsAny<int?>(),
            It.IsAny<CommandType?>()
        )).ReturnsAsync(expectedUser);

        // 执行
        await _repository.GetByIdAsync(1);

        // 断言:验证调用的SQL和参数
        _mockConnection.Verify(c => c.QueryFirstOrDefaultAsync<User>(
            "SELECT Id, Name, Age FROM User WHERE Id = @Id",
            It.Is<object>(p => ((dynamic)p).Id == 1),
            null,
            null,
            null
        ), Times.Once);
    }

    [TestMethod]
    public async Task AddAsync_CallsCorrectSqlAndParameters()
    {
        // 准备
        var newId = 5;
        _mockConnection.Setup(c => c.ExecuteScalarAsync<int>(
            It.IsAny<string>(),
            It.IsAny<object>(),
            It.IsAny<IDbTransaction>(),
            It.IsAny<int?>(),
            It.IsAny<CommandType?>()
        )).ReturnsAsync(newId);

        var user = new User { Name = "王五", Age = 28 };

        // 执行
        var result = await _repository.AddAsync(user);

        // 断言
        Assert.AreEqual(newId, result);
        _mockConnection.Verify(c => c.ExecuteScalarAsync<int>(
            "INSERT INTO User (Name, Age) VALUES (@Name, @Age); SELECT last_insert_rowid()",
            It.Is<object>(p => ((dynamic)p).Name == "王五" && ((dynamic)p).Age == 28),
            null,
            null,
            null
        ), Times.Once);
    }
}

两种方案的适用场景

  • 内存数据库方案适合需要验证SQL语句逻辑正确性的场景,能够更真实地模拟数据操作流程,测试覆盖更贴近实际执行情况。
  • Mock框架方案适合只需要验证仓储层调用参数的场景,测试执行速度更快,不需要启动数据库环境,适合对调用逻辑做快速验证。

在实际项目中,可以根据测试目标选择合适的方案,也可以两种方案结合使用,既保证SQL逻辑的正确性,也验证调用参数的准确性。

Dapper单元测试仓储层SQL_Mapper修改时间:2026-07-04 19:09:22

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。