我正在创建一个应用程序,它将管理外部实用程序的多个实例,为每个实例提供数据并获取结果.
但是当我为班级编写单元测试时,我遇到了一个问题.
如何在调用时测试目标方法实际启动进程(通过属性设置)?
我试过了:
使该类执行外部进程,然后使用GetProcessesByName检查它是否已启动.
使用输出重定向,例如使用大于号来回显文件并测试其存在
我觉得发出代码和/或创建另一个exe来测试是矫枉过正.
这是确切的方法:
public void Start() { if (!_isRunning) { var startInfo = new ProcessStartInfo() { CreateNoWindow = true, UseShellExecute = true, FileName = _cmdLine, Arguments = _args }; _process = Process.Start(startInfo); _isRunning = true; } else { throw new InvalidOperationException("Process already started"); } }
我想对它进行单元测试,这样如果没有运行(_isRunning == false),就应该生成一个新进程.
我感到难过,是否有一种优雅的方式来对外部过程实际启动进行单元测试?
我会使用依赖注入和使用模拟或假类来解决这个问题.注意我使用实例方法来启动而不是类方法.在常规代码中,您可以使用默认构造函数,它将为您创建一个流程.对于测试,您可以注入模拟或伪造进程,只需检查在模拟对象上调用正确的方法,而不必实际启动进程.您需要调整此项以考虑我省略的属性.防爆.下面:
public class UtilityManager { public Process UtilityProcess { get; private set; } private bool _isRunning; public UtilityManager() : this(null) {} public UtilityManager( Process process ) { this. UtilityProcess = process ?? new Process(); this._isRunning = false; } public void Start() { if (!_isRunning) { var startInfo = new ProcessStartInfo() { CreateNoWindow = true, UseShellExecute = true, FileName = _cmdLine, Arguments = _args }; this.UtilityProcess.Start(startInfo); _isRunning = true; } else { throw new InvalidOperationException("Process already started"); } }
测试代码......
[TestMethod] public void StartTest() { Process proc = new FakeProcess(); // May need to use a wrapper class UtilityManager manager = new UtilityManager( proc ); manager.CommandLine = "command"; ... manager.Start(); Assert.IsTrue( proc.StartCalled ); Assert.IsNotNull( proc.StartInfo ); Assert.AreEqual( "command", proc.StartInfo.FileName ); ... }