C# 如何在windows窗体代码的后面编写方法的单元测试

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)

我有一个我想要测试的方法,我得到一个错误“在创建窗口句柄之前不能对控件调用Invoke或BeginInvoke”。现在我还有一个列表框,它正在我测试的函数中填充。因此,当我将该方法分离到另一个类时,这是一个问题

我理解这一点是因为表单需要首先运行,但是有其他选择吗

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和逻辑)。检查并确保您阅读了我添加的替代解决方案部分。