C# 如何在windows窗体代码的后面编写方法的单元测试
我有一个我想要测试的方法,我得到一个错误“在创建窗口句柄之前不能对控件调用Invoke或BeginInvoke”。现在我还有一个列表框,它正在我测试的函数中填充。因此,当我将该方法分离到另一个类时,这是一个问题 我理解这一点是因为表单需要首先运行,但是有其他选择吗C# 如何在windows窗体代码的后面编写方法的单元测试,c#,.net,winforms,unit-testing,mstest,C#,.net,Winforms,Unit Testing,Mstest,我有一个我想要测试的方法,我得到一个错误“在创建窗口句柄之前不能对控件调用Invoke或BeginInvoke”。现在我还有一个列表框,它正在我测试的函数中填充。因此,当我将该方法分离到另一个类时,这是一个问题 我理解这一点是因为表单需要首先运行,但是有其他选择吗 public partial class ImportForm : Form { public ImportForm() { } public bool Test(string[] fileNames)
public partial class ImportForm : Form
{
public ImportForm()
{
}
public bool Test(string[] fileNames)//Method to test
{
foreach (DataTable table in result.Tables)
{
foreach (DataRow dr in table.Rows)
{
if (!db.CouncilRefundCases.Any(
c => c.RequestReference == dr.ItemArray[1].ToString()))
{
CouncilRefundCase data = new CouncilRefundCase()
{
FileId = fileId,
RequestReference = Convert.ToString(dr.ItemArray[1]),
CancelReason = Convert.ToString(dr.ItemArray[2]),
ProcessStatusId = (int?)ProcessStatus.Unprocessed,
ProcessDescription = new EnumHelper().GetDescription(ProcessStatus.Unprocessed),
DateCaptured = DateTime.Now
};
db.CouncilRefundCases.InsertOnSubmit(data);
//Succeeded ones
var item = new ListViewItem(dr.ItemArray[1].ToString());
lstSuccessSummary.Invoke((Action)delegate
{
lstSuccessSummary.Items.Add(item);
});
}
else
{
//Failed ones
var item = new ListViewItem(dr.ItemArray[1].ToString());
lstSummary.Invoke((Action)delegate
{
lstSummary.Items.Add(item);
});
}
}
}
return true;
}
}
这是我的单元测试
[TestMethod]
public void TestTest()
{
bool results=false;
var files = new string[4];
files[0] = @"filename1.xlsx";
files[1] = @"filename2.xlsx";
ImportForm form= new ImportForm();
results = form.Test(files);
Assert.AreEqual(true, results);
}
如果没有实际的代码很难判断,但通常这表明
Test
方法中的代码不应该在表单中
表单实际上应该只是显示逻辑,而不是模型逻辑
现在,我还有一个列表框,正在该列表框中填充
我正在测试的功能。因此,当我将方法分离到
另一节课
如果您用一些代码描述这个问题,我们可能会提供帮助。您可以使用事件或委托解决问题,以避免视图逻辑出现在模型之外,反之亦然。注意-在阅读答案之前 一般来说,将UI代码和业务逻辑紧密耦合不是一个好主意,但如果您遇到无法重构为与UI解耦的代码,您可以使用以下解决方案来解决问题 问题和解决方案 在显示窗体之前,窗体及其控件未处于状态,您不能使用窗体或其控件的
Invoke
方法
要解决此问题,可以强制创建窗体及其控件。为此,调用表单的内部方法并将true
作为参数传递给它就足够了:
var f = new Form1();
var createControl = f.GetType().GetMethod("CreateControl",
BindingFlags.Instance | BindingFlags.NonPublic);
createControl.Invoke(f, new object[] { true });
替代解决方案
- 在调用方法之前显示
。然后,该表单将在运行单元测试时显示表单
- 在
线程中显示STA
表单
public partial class Form1 : Form
{
//...
public int Method1(int i)
{
this.Invoke(new Action(() => { i++; }));
return i;
}
}
然后在测试项目中,可以使用以下代码:
[TestMethod]
public void TestMethod1()
{
var f = new Form1();
var createControl = f.GetType().GetMethod("CreateControl",
BindingFlags.Instance | BindingFlags.NonPublic);
createControl.Invoke(f, new object[] { true });
var input = 0;
var expected = 1;
var actual = f.Method1(input);
Assert.AreEqual(expected, actual);
}
相关:虽然我会建议将逻辑与Winforms分离以进行更好的测试,但您仍然可以进行良好的测试,这将测试整个应用程序(UI和逻辑)。检查并确保您阅读了我添加的替代解决方案部分。