3

主应用程序窗口,减去大量不相关的代码。

<Window>
    <Window.Resources>
    <DataTemplate DataType="{x:Type presenters:DashboardViewModel}">
        <views:DashboardView />
    </DataTemplate>
    <DataTemplate DataType="{x:Type presenters:SecondViewModel}">
        <views:SecondView />
    </DataTemplate>
    </Window.Resources>

    <ContentPresenter Content="{Binding WindowPresenter}"/>
</Window>

绑定到 Window 的视图模型

public class RootViewModel {
    // IRL this implements notifypropchanged
    public IPresenter WindowPresenter {get; set;}
    public void ShowDashboard(){ this.WindowPresenter = new DashBoardViewModel(); }
    public void ShowSecond(){ this.WindowPresenter = new SecondViewModel(); }
}

DashboardView并且SecondView是具有许多依赖属性的用户控件,这些依赖属性绑定到各自视图模型中的属性。

// example of a common dependency property I have 
public static readonly DependencyProperty ColorPaletteProperty = DependencyProperty.Register("ColorPalette", typeof(ColorPalette), typeof(SurfaceMapControl), new PropertyMetadata(ColorPalette.Rainbow, new PropertyChangedCallback(SurfaceMapControl.ColorPalettePropertyChanged)));
private static void ColorPalettePropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) { ((SurfaceMapControl)d).OnColorRangeChanged(); }
private void OnColorRangeChanged() { 
    // code that uses this.SomeOtherDependencyProperty
    // throws null ref exception
}

调用ShowDashboard()时,内容呈现器显示正确的用户控件并且所有属性都正确绑定。在调用 时ShowSecond(),内容呈现器显示正确的用户控件并且所有属性都正确绑定。

有时,在两个视图之间切换时,我会在其中一个用户控件的依赖属性中得到空引用异常,因为我的一些属性会查看其他依赖属性。这让我相信视图模型在视图之前被垃圾收集,并且视图模型中的更改触发了 usercontrols 依赖属性,这反过来又引发了异常,因为视图模型不再存在。

我可以防止在视图模型被处置时触发依赖属性吗?

或者是否有必要在每个依赖属性中进行空数据上下文检查?

我应该在这里包含一些东西来查看用户控件的生命周期以完全防止这种情况吗?

4

1 回答 1

1

WPF 在绑定时通常对所有绑定操作使用弱引用,以防止发生内存泄漏。

因此,您ViewModel可能会在某个时候被清理并消失,而控件仍然是“活动的”,因为 ViewModel 上的 GC 可能在视图实际关闭之前发生。

最简单的解决方案通常是通过空检查处理这些更改通知,并跳过代码的适当部分。这也很有用,具体取决于您在初始化/创建时的设置方式。

于 2013-11-05T20:43:02.763 回答