geo*_*tnz 5 silverlight events mvvm
单行摘要:解除在Silverlight2中UserControl的构造函数中创建的事件处理程序的最佳实践是什么?
背景: 我目前正在Silverlight2中构建业务线应用程序.由于Silverlight是一个浏览器插件,因此没有Window的概念 - 一切都在UserControls中完成.我在应用程序中处理不同"表单"的方式是使用包含Viewbox的顶级用户控件.为了显示不同的表单,我将Viewbox的Child属性设置为不同的UserControls.我的应用程序有一个单独的PageManager类,可以调用它来打开和关闭表单.表单(UserControls)存储在堆栈中.打开一个表单将它放在堆栈的顶部,关闭它将其从堆栈中删除并显示它下面的一个.
我正在尝试遵循Model-View-ViewModel模式.在每种形式(从UserControl派生)中,我有一个ViewModel来管理View的所有数据.ViewModel公开事件,以便在完成加载和保存等操作时通知UI.
在我的表单中,在我获得ViewModel之后,我在构造函数中订阅了该事件
public partial class MyPage : UserControl
{
public MyViewModel ViewModel{get; set;}
// other constructors, which create the viewmodel and call the constructor below.
public MyPage(MyViewModel viewModel)
{
InitializeComponent();
ViewModel = viewModel;
this.LayoutRoot.DataContext = this.ViewModel;
// subscribe to event so we can do stuff
this.ViewModel.LoadCompleted += new MyViewModel.LoadCompletedEventHandler(ViewModel_LoadCompleted);
}
Run Code Online (Sandbox Code Playgroud)
我的问题是:现在我订阅了这个事件,我何时删除处理程序?我是否创建了一个析构函数并在那里创建,或者是否会创建一个鸡与蛋的情况,垃圾收集器不会破坏对象直到所有引用(即:事件处理程序)都消失了?我是否创建了一个表单必须实现的接口,该接口指定在PageManager关闭表单时调用的UnhookEvents函数?
编辑:感谢您的回复.ViewModel的持续时间比表单(UserControl)长的情况怎么样?我的应用程序的一部分允许用户创建一个非常复杂的结构,但在95%的情况下,它更简单.我所做的是创建两个使用相同ViewModel的表单.用户可以开始填写简单表单,然后切换到高级模式,这将创建一个新表单,将ViewModel传递给它.
在简单的设置形式:
private void AdvancedSessionSetupButton_Click(object sender, RoutedEventArgs e)
{
PageManager.GetPageManager().Close(this);
PageManager.GetPageManager().Open(new CreateSessionPage(this.ViewModel), "Create Session");
}
Run Code Online (Sandbox Code Playgroud)
在高级设置表单中:
private void BasicSessionSetupButton_Click(object sender, RoutedEventArgs e)
{
PageManager.GetPageManager().Close(this);
PageManager.GetPageManager().Open(new CreateBasicSessionPage(this.ViewModel), "Create Session");
}
Run Code Online (Sandbox Code Playgroud)
在PageManager.Close之后,引用表单的唯一内容是ViewModel中的事件.我想这就是我应该解开他们的地方.
在这种情况下,不需要析构函数(C# 程序员通常称为终结器)。假设 ViewModel_LoadCompleted 是一个成员函数,它包含一个指向“this”的指针,您将其提供给完全包含在“this”中的 ViewModel 对象。垃圾收集器应该明智地忽略这一点。
在这种情况下,正确的做法是不要浪费时间解绑它们。
一般来说,当您将“this”(显式或隐式)传递给某个对象时,您需要取消绑定事件处理程序,该对象将保留该引用的时间长于“this”的预期生存期。例如,如果您在父控件的事件上设置处理程序。现在,父级可以通过处理程序及其子控件集合对您进行引用。在这种情况下,当您从父级中移除时,您应该取消绑定。
如有疑问,请在调用 Dispose() 时实现 IDisposable 并取消绑定。