断言已使用 xUnit 调用该方法

Que*_*n3r 3 c# unit-testing xunit

我有一个带有我想测试的日志记录方法的类。对于这个例子,我想检查该Console.WriteLine方法是否已被调用。这是我的示例课程

public class MyClass
{
    public void LogSomething()
    {
        Console.WriteLine("Test");
    }
}
Run Code Online (Sandbox Code Playgroud)

及其测试

public class MyClassTests
{
    [Fact]
    public void LogsSomething()
    {
        MyClass myClass = new MyClass();
            
        myClass.LogSomething(); // Assert that Console.WriteLine has been called once
    }
}
Run Code Online (Sandbox Code Playgroud)

有什么我可以用的吗?(最好没有额外的包)

我正在寻找这样的断言(伪代码)

  • Assert.Method(Console.WriteLine).ToHaveBeenCalledWith(myClass.LogSomething);
  • Assert.Method(Console.WriteLine).ToHaveBeenCalledWith(myClass.LogSomething).Times(3); // Check if Console.WriteLine has been called 3 times (loop inside the LogSomething method)

Rod*_*rez 5

我认为你不能立即断言这一点。

您最好的选择是使用Moq或其他模拟框架来执行此操作。您应该始终以使用依赖项注入的解耦逻辑为目标,否则您最终会得到一个紧密耦合的代码,该代码难以测试,并且每当需求发生变化时都不容易重构

public interface ILogger
{
    public void Log();    
}
Run Code Online (Sandbox Code Playgroud)
public class Logger: ILogger
{
    public void Log()
    {
        Console.WriteLine("Look mom, i'm logging");
    }
}
Run Code Online (Sandbox Code Playgroud)
public class MyClass
{
    private readonly ILogger Logger
    
    public MyClass(ILogger logger)
    {
        Logger = logger;
    }

    public void MyMethod()
    {
        //other code
        Logger.Log();
    }
}
Run Code Online (Sandbox Code Playgroud)
public class MyClassTests
{
    [Fact]
    public void LogsSomething()
    {
      //arrange
      var logger = new Mock<ILogger>();   
      
      //act
      var sut = new MyClass(logger.Object);
      sut.MyMethod();

      //Assert 
      logger.Verify(foo => foo.Log(), Times.Once()); //here
      //some other assertions
    }
}
Run Code Online (Sandbox Code Playgroud)