C# 如何进行单元测试 - MSTest, xUnit, NUnit框架入门

MSTest、xUnit、NUnit是C#主流单元测试框架,差异在于定位与适用场景:MSTest适合VS集成企业项目,xUnit强调无状态测试设计,NUnit以数据驱动和生态丰富见长;选型应基于团队技术栈与项目需求。

单元测试是保障C#代码质量的关键环节,选对框架、写好断言、组织好测试结构,比“有没有测”更重要。MSTest、xUnit 和 NUnit 是目前最主流的三大框架,它们目标一致,但设计哲学和用法细节有明显差异。

先搞清三者的定位差异

不是“哪个更好”,而是“哪个更适合当前场景”:

  • MSTest:微软官方出品,深度集成 Visual Studio,适合企业级 .NET 项目、特别是使用 Azure DevOps 或需要 Test Explorer 图形化运行的团队;语法简洁,但扩展性相对保守。
  • xUnit:强调“约定优于配置”,默认禁用测试类实例重用(每个测试方法独享新实例),强制你写无状态、高内聚的测试;不支持 `[ExpectedException]` 这类老式异常断言,推荐用 `Assert.Throws()`,更现代也更严谨。
  • NUnit:历史最久、文档最全、特性最丰富(如 `[TestCase]`、`[Values]`、`[Theory]` + `[TestCaseSource]` 等数据驱动能力极强);语法灵活,学习曲线平缓,

    适合从入门到进阶长期使用。

快速上手:一个真实例子跑通三框架

假设你有一个简单计算器类:

public class Calculator { public int Add(int a, int b) => a + b; }

下面分别是三个框架中对应的一个测试方法写法(均在测试项目中):

  • MSTest
    [TestMethod]
    public void Add_ReturnsCorrectSum() {
      var calc = new Calculator();
      Assert.AreEqual(5, calc.Add(2, 3));
    }
  • xUnit
    [Fact]
    public void Add_ReturnsCorrectSum() {
      var calc = new Calculator();
      Assert.Equal(5, calc.Add(2, 3));
    }
  • NUnit
    [Test]
    public void Add_ReturnsCorrectSum() {
      var calc = new Calculator();
      Assert.That(calc.Add(2, 3), Is.EqualTo(5));
    }

注意:三者都需要安装对应 NuGet 包(MSTest.TestFrameworkxunit + xunit.runner.visualstudioNUnit + NUnit3TestAdapter),且测试项目 SDK 类型建议用 Microsoft.NET.Test.Sdk

关键实践建议,避开常见坑

  • 测试命名要表达意图,别叫 TestAdd,推荐 Add_TwoPositiveNumbers_ReturnsCorrectSum —— 框架不关心名字,但人关心,尤其后期排查时。
  • 一个测试只验证一个关注点:不要在一个 `[Fact]` 或 `[Test]` 里断言多个逻辑分支;用多个独立测试覆盖不同输入(比如 Add(-1, 1)、Add(0, 0)、Add(int.MaxValue, 1) 触发溢出等)。
  • 善用生命周期管理:MSTest 用 `[ClassInitialize]/[TestInitialize]`,xUnit 用构造函数 + `IDisposable`,NUnit 用 `[OneTimeSetUp]/[SetUp]` —— 但优先考虑“每个测试干净独立”,而非过度复用 setup 逻辑。
  • 异步测试必须显式标记:xUnit 用 `public async Task MyTest()` + `await`;MSTest 用 `[TestMethod] public async Task`;NUnit 用 `[Test] public async Task` —— 否则可能误判为同步完成而跳过等待。

怎么选?看团队和场景

如果你刚入门,或团队已用 Visual Studio 做 CI/CD,MSTest 上手最快;如果追求测试设计的纯粹性和未来可维护性,xUnit 是很好的选择;如果项目需要大量参数化测试、兼容老项目、或希望生态工具链(如 ReportGenerator、NUnitLite)更丰富,NUnit 更稳。

基本上就这些 —— 框架只是工具,真正起作用的是你写的每一个有意义的断言、每一个隔离良好的测试上下文。