理想情况下,您希望将表单注册为Singleton
。然而,根据我的经验,这将导致难以调试的错误,特别是当您使用BindingSource https://msdn.microsoft.com/en-us/library/system.windows.forms.bindingsource.datasource(v=vs.110).aspx用于将您的数据绑定到任何内容。
使用中的第二个问题Singleton
因为生活方式是,如果您的应用程序使用无模式窗口,该窗口将抛出一个ObjectDisposedException
第二次打开时,因为 Windows 窗体应用程序框架将在第一次关闭时处理该窗体,而简单注入器应负责该操作。因此,如果注册为 Singleton,Simple Injector 将创建一个(而且恰好是一个)实例。如果其他人(例如您的应用程序、Windows 窗体框架)将处置该对象,则不会重新创建该对象。
最简单的解决方案也很容易理解,是将您的表单注册为Transient
。是的,您需要抑制诊断警告。根据该诊断警告的原因文档 https://simpleinjector.org/diadt:
一个实现的组件IDisposable
通常需要确定性清理,但简单注入器不会隐式跟踪和处置使用瞬态生活方式注册的组件。
简单注入器无法处置瞬态组件,因为它无法确定何时应处置对象。然而,这意味着通过调用以模式方式打开的表单.ShowDialog()
永远不会被处置!而且由于 Windows 窗体应用程序通常会运行很长时间,甚至可能是一周或一个月,因此这最终会导致'Win32异常' https://msdn.microsoft.com/en-us/library/system.componentmodel.win32exception(v=vs.110).aspx并显示一条消息:“创建窗口句柄时出错”。这本质上意味着您耗尽了计算机的所有资源。
因此,处理表格很重要。尽管简单注射器能够完成这项工作,但如果您使用Scope https://simpleinjector.org/lifestyles#scoped,这对于 Windows Forms 来说不是那么容易实现的。因此,您自己必须负责处理已使用显示的封闭表格ShowDialog()
.
根据您的具体用例,有多种方法可以实现FormOpener
or NavigationService
。一种方法是:
public interface IFormOpener
{
void ShowModelessForm<TForm>() where TForm : Form;
DialogResult ShowModalForm<TForm>() where TForm : Form;
}
public class FormOpener : IFormOpener
{
private readonly Container container;
private readonly Dictionary<Type, Form> openedForms;
public FormOpener(Container container)
{
this.container = container;
this.openedForms = new Dictionary<Type, Form>();
}
public void ShowModelessForm<TForm>() where TForm : Form
{
Form form;
if (this.openedForms.ContainsKey(typeof(TForm)))
{
// a form can be held open in the background, somewhat like
// singleton behavior, and reopened/reshown this way
// when a form is 'closed' using form.Hide()
form = this.openedForms[typeof(TForm)];
}
else
{
form = this.GetForm<TForm>();
this.openedForms.Add(form.GetType(), form);
// the form will be closed and disposed when form.Closed is called
// Remove it from the cached instances so it can be recreated
form.Closed += (s, e) => this.openedForms.Remove(form.GetType());
}
form.Show();
}
public DialogResult ShowModalForm<TForm>() where TForm : Form
{
using (var form = this.GetForm<TForm>())
{
return form.ShowDialog();
}
}
private Form GetForm<TForm>() where TForm : Form
{
return this.container.GetInstance<TForm>();
}
}
该类必须注册为Singleton
:
container.RegisterSingleton<IFormOpener, FormOpener>();
并且可以通过将此服务注入到应用程序的根表单中来使用:
public partial class RootForm : Form
{
private readonly IFormOpener formOpener;
public RootForm(IFormOpener formOpener)
{
this.formOpener = formOpener;
this.InitializeComponent();
}
private void ShowCustomers_Click(object sender, EventArgs e)
{
this.formOpener.ShowModelessForm<AllCustomersForm>();
}
private void EditCustomer_Click(object sender, EventArgs e)
{
var result = this.formOpener.ShowModalForm<EditCustomerForm>();
// do something with result
}
}