首页 > 解决方案 > 如何在windows表单代码后面为方法编写单元测试

问题描述

我有要测试的方法,但我收到此错误“在创建窗口句柄之前无法在控件上调用 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);
}

标签: c#.netwinformsunit-testingmstest

解决方案


注意 - 在您阅读答案之前

一般来说,将 UI 代码和业务逻辑紧密耦合并不是一个好主意,但是如果您遇到无法重构以与 UI 解耦的代码,您可以使用以下解决方案来解决问题。

问题和解决方案

在显示窗体之前,窗体及其控件未处于Created状态,您不能使用Invoke窗体或其控件的方法。

要解决此问题,您可以强制创建表单及其控件。为此,只需调用表单的内部方法并将其作为参数CreateControl(bool fIgnoreVisible)传递给它:true

var f = new Form1();
var createControl = f.GetType().GetMethod("CreateControl",
    BindingFlags.Instance | BindingFlags.NonPublic);
createControl.Invoke(f, new object[] { true });

替代解决方案

  • Form在调用方法之前显示。然后表单将在运行单元测试期间显示。

  • FormSTA线程中显示。

例子

假设您的表单中有这样的方法:

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);
}

推荐阅读