在C#中模拟StreamReader以进行单元测试

Ale*_*lex 3 c# unit-testing moq

我有这种格式的代码:

public void parseFile(string filePath)
{
    using (var reader = new StreamReader(@filePath))
    {
        //Do something
    }
}
Run Code Online (Sandbox Code Playgroud)

现在,我想对代码进行单元测试,但不希望单元测试实际访问文件系统,因为如果实际文件不存在,这会使它们不可靠。

为了防止streamreader访问文件系统,我需要对其进行模拟。我的理解是,我只能模拟实现接口的类。

因此,我可以看到的一个解决方案是为制作包装类和接口Streamreader,然后可以对其进行模拟。

我的问题分为两个部分:

  1. 这是解决此问题的唯一方法吗?

  2. 为了方便进行单元测试,在项目中添加一个额外的层是否正确?如果我在全球范围内遵循这一原则,我会增加很多额外的课程吗?

Ayd*_*din 7

上面的示例的问题在于该parseFile方法正在创建自己的实例StreamReader,因此模拟StreamReader不会真正起作用,因为:

  1. 您无权访问该对象。
  2. 您只能模拟interfaces或标记为的类的成员virtual

相反,您可以做的是创建一个接口,IFileManager为方便起见,我们使用名为的方法来调用它StreamReader

public interface IFileManager
{
     StreamReader StreamReader(string path);
}
Run Code Online (Sandbox Code Playgroud)

然后在另一个类(我们称之为Foo)中,该类包含ParseFile您上面发布的方法:

public class Foo 
{
    IFileManager fileManager;

    public Foo(IFileManager fileManager)
    {
        this.fileManager = fileManager;
    }

    public void parseFile(string filePath)
    {
        using (var reader = fileManager.StreamReader(filePath))
        {
            //Do something
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

现在,您可以模拟IFileManager interface和其StreamReader方法,可以将此模拟的实例注入到Foo类中,以供ParseFile方法使用。

现在您的代码将依赖于抽象而不是具体的实现,我们已经反转了依赖关系,这使我们可以模拟依赖关系并隔离我们要实际测试的代码。


创建模拟对象的粗略演示

public static void Main()
{
    Mock<IFileManager> mockFileManager = new Mock<IFileManager>();
    string fakeFileContents = "Hello world";
    byte[] fakeFileBytes = Encoding.UTF8.GetBytes(fakeFileContents);

    MemoryStream fakeMemoryStream = new MemoryStream(fakeFileBytes);

    mockFileManager.Setup(fileManager => fileManager.StreamReader(It.IsAny<string>()))
                   .Returns(() => new StreamReader(fakeMemoryStream));

    Foo foo = new Foo(mockFileManager.Object);

    string result = foo.ParseFile("test.txt");

    Console.WriteLine(result);
}

public interface IFileManager
{
    StreamReader StreamReader(string path);
}

public class Foo
{
    IFileManager fileManager;

    public Foo(IFileManager fileManager)
    {
        this.fileManager = fileManager;
    }

    public string ParseFile(string filePath)
    {
        using (var reader = fileManager.StreamReader(filePath))
        {
            return reader.ReadToEnd();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)