C#如何使用xUnit?xUnit完整教程与实战案例详解

来源:网站主作者:木下头衔:网络博主
导读:本期聚焦于小伙伴创作的《C#如何使用xUnit?xUnit完整教程与实战案例详解》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《C#如何使用xUnit?xUnit完整教程与实战案例详解》有用,将其分享出去将是对创作者最好的鼓励。

在C#项目开发过程中,单元测试是验证代码逻辑正确性、减少线上问题的重要手段,xUnit是.NET生态中主流的单元测试框架之一,具有轻量、扩展性强、运行速度快的特点,很多开发者会选择它来完成项目的测试工作。

C#如何使用xUnit?xUnit完整教程与实战案例详解

xUnit环境准备

首先需要在项目中安装xUnit相关的NuGet包,假设我们有一个待测试的C#类库项目,步骤如下:

  • 在解决方案资源管理器中右键点击解决方案,选择添加新项目
  • 选择xUnit测试项目模板,命名为Demo_Test
  • 项目创建完成后,会自动引用xUnit的核心包,同时需要手动添加待测试项目的引用

安装完成后,项目的csproj文件会包含如下依赖配置:

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <TargetFramework>net8.0</TargetFramework>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.8.0" />
    <PackageReference Include="xunit" Version="2.6.2" />
    <PackageReference Include="xunit.runner.visualstudio" Version="2.5.4" />
    <ProjectReference Include="..DemoDemo.csproj" />
  </ItemGroup>

</Project>

编写第一个xUnit测试用例

假设待测试的项目中有一个简单的计算器类,代码如下:

namespace Demo;

public class Calculator
{
    // 加法计算
    public int Add(int a, int b)
    {
        return a + b;
    }

    // 除法计算
    public int Divide(int a, int b)
    {
        if (b == 0)
        {
            throw new ArgumentException("除数不能为0");
        }
        return a / b;
    }
}

接下来在测试项目中编写对应的测试用例,xUnit的测试方法不需要添加额外的前缀,只需要标记[Fact]特性即可:

using Demo;
using Xunit;

namespace Demo_Test;

public class CalculatorTest
{
    private readonly Calculator _calculator;

    public CalculatorTest()
    {
        _calculator = new Calculator();
    }

    [Fact]
    public void Add_ShouldReturnCorrectSum()
    {
        // 准备测试数据
        int a = 3;
        int b = 5;
        // 执行测试方法
        int result = _calculator.Add(a, b);
        // 断言结果
        Assert.Equal(8, result);
    }
}

运行测试后,如果结果符合预期,测试会显示通过状态。

xUnit常用断言方法

断言是测试用例的核心部分,xUnit提供了丰富的断言方法,常用的包括:

  • Assert.Equal(expected, actual):判断两个值是否相等
  • Assert.NotEqual(notExpected, actual):判断两个值是否不相等
  • Assert.True(condition):判断条件是否为真
  • Assert.False(condition):判断条件是否为假
  • Assert.Throws<TException>(() => ...):判断是否会抛出指定类型的异常
  • Assert.Contains(expectedSubstring, actualString):判断字符串是否包含指定子串

我们可以为计算器的除法方法编写测试用例,验证异常抛出场景:

[Fact]
public void Divide_WhenDivisorIsZero_ShouldThrowArgumentException()
{
    // 准备测试数据
    int a = 10;
    int b = 0;
    // 断言会抛出ArgumentException异常
    var exception = Assert.Throws<ArgumentException>(() => _calculator.Divide(a, b));
    // 验证异常消息是否符合预期
    Assert.Equal("除数不能为0", exception.Message);
}

使用[Theory]实现参数化测试

如果需要对同一个测试方法传入多组不同的参数进行验证,可以使用[Theory]特性配合[InlineData]来完成参数化测试,避免重复编写类似的测试方法:

[Theory]
[InlineData(1, 2, 3)]
[InlineData(-1, 5, 4)]
[InlineData(0, 0, 0)]
[InlineData(100, 200, 300)]
public void Add_ParameterizedTest_ShouldReturnCorrectSum(int a, int b, int expected)
{
    int result = _calculator.Add(a, b);
    Assert.Equal(expected, result);
}

运行这个测试方法时,xUnit会自动对每一组[InlineData]的数据执行一次测试,只要有一组数据不符合预期,整个测试方法就会显示失败。

测试分组与跳过测试

可以通过[Trait]特性给测试用例添加分组标签,方便后续按分组运行测试:

[Fact]
[Trait("Category", "Basic")]
public void Add_BasicCategory_ShouldReturnCorrectSum()
{
    int result = _calculator.Add(2, 3);
    Assert.Equal(5, result);
}

如果某个测试用例暂时不需要运行,可以使用[Fact(Skip = "跳过原因")]来跳过该测试:

[Fact(Skip = "该功能尚未开发完成,暂时跳过测试")]
public void Multiply_ShouldReturnCorrectResult()
{
    // 待实现的计算器乘法方法测试
}

完整实战案例:用户服务测试

假设我们有一个用户服务类,提供用户查询和密码验证的功能,代码如下:

namespace Demo;

public class User
{
    public int Id { get; set; }
    public string UserName { get; set; }
    public string PasswordHash { get; set; }
}

public class UserService
{
    private readonly List<User> _users;

    public UserService()
    {
        // 模拟数据库中的用户数据
        _users = new List<User>
        {
            new User { Id = 1, UserName = "admin", PasswordHash = "123456" },
            new User { Id = 2, UserName = "test", PasswordHash = "abcdef" }
        };
    }

    // 根据用户名查询用户
    public User GetUserByUserName(string userName)
    {
        if (string.IsNullOrEmpty(userName))
        {
            throw new ArgumentException("用户名不能为空");
        }
        return _users.FirstOrDefault(u => u.UserName == userName);
    }

    // 验证用户密码
    public bool ValidatePassword(string userName, string password)
    {
        var user = GetUserByUserName(userName);
        if (user == null)
        {
            return false;
        }
        return user.PasswordHash == password;
    }
}

对应的xUnit测试用例如下:

using Demo;
using Xunit;

namespace Demo_Test;

public class UserServiceTest
{
    private readonly UserService _userService;

    public UserServiceTest()
    {
        _userService = new UserService();
    }

    [Fact]
    public void GetUserByUserName_ValidUserName_ShouldReturnCorrectUser()
    {
        var user = _userService.GetUserByUserName("admin");
        Assert.NotNull(user);
        Assert.Equal(1, user.Id);
        Assert.Equal("admin", user.UserName);
    }

    [Fact]
    public void GetUserByUserName_InvalidUserName_ShouldReturnNull()
    {
        var user = _userService.GetUserByUserName("nonexistent");
        Assert.Null(user);
    }

    [Fact]
    public void GetUserByUserName_EmptyUserName_ShouldThrowArgumentException()
    {
        Assert.Throws<ArgumentException>(() => _userService.GetUserByUserName(""));
    }

    [Theory]
    [InlineData("admin", "123456", true)]
    [InlineData("admin", "wrongpassword", false)]
    [InlineData("test", "abcdef", true)]
    [InlineData("nonexistent", "123456", false)]
    public void ValidatePassword_ShouldReturnCorrectResult(string userName, string password, bool expected)
    {
        bool result = _userService.ValidatePassword(userName, password);
        Assert.Equal(expected, result);
    }
}

运行所有测试用例后,可以查看每个测试的执行结果,确保所有逻辑都符合预期,后续如果修改了用户服务的代码,只需要重新运行测试就可以快速验证是否引入了新的问题。

xUnitC#单元测试测试框架修改时间:2026-06-20 10:21:43

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