21

我对 WPF 比较陌生,它的一些东西对我来说很陌生。一方面,与 Windows 窗体不同,WPF 控件层次结构不支持 IDisposable。在 Windows 窗体中,如果用户控件使用任何托管资源,则通过重写每个控件实现的 Dispose 方法来清理资源非常容易。

在 WPF 中,事情并没有那么简单。我已经搜索了几个小时,并且遇到了两个基本主题:

第一个主题是 Microsoft 明确指出 WPF 不实现 IDisposable,因为 WPF 控件没有非托管资源。虽然这可能是真的,但他们似乎完全忽略了这样一个事实,即用户对其 WPF 类层次结构的扩展可能确实使用托管资源(直接或间接通过模型)。通过不实现 IDisposable,Microsoft 有效地删除了唯一有保证的机​​制,通过该机制可以清理自定义 WPF 控件或窗口使用的非托管资源。

其次,我发现了一些对 Dispatcher.ShutdownStarted 的引用。我曾尝试使用 ShutdownStarted 事件,但它似乎并没有为每个控件触发。我有一堆 WPF UserControl,我已经为 ShutdownStarted 实现了一个处理程序,并且它永远不会被调用。我不确定它是否仅适用于 Windows,或者 WPF App 类。但是它没有正确触发,每次应用程序关闭时我都会泄漏打开的 PerformanceCounter 对象。

有没有比 Dispatcher.ShutdownStarted 事件更好的方法来清理非托管资源?实现 IDisposable 是否有一些技巧可以调用 Dispose ?如果可能的话,我更愿意避免使用终结器。

4

5 回答 5

13

恐怕 Dispatcher.ShutdownStarted 似乎确实是 WPF 提供的用于在 UserControls 中处理资源的唯一机制。(参见我不久前提出的一个非常相似的问题)。

解决该问题的另一种方法是将所有可支配资源(如果可能的话)从背后的代码中移出并放入单独的类中(例如使用 MVVM 模式时的 ViewModel)。然后在更高级别上,您可以处理主窗口关闭并通过 Messenger 类通知所有 ViewModel。

我很惊讶您没有收到 Dispatcher.ShutdownStarted 事件。您的用户控件当时是否附加到顶级窗口?

于 2009-10-11T13:16:59.573 回答
11

IDisposable 接口在 WPF 下(几乎)没有意义,因为该机制与 Winforms 不同。在 WPF 中,您必须牢记视觉和逻辑树:这是基本的。
因此,任何视觉对象通常都作为其他对象的子对象存在。WPF 构建机制的基础是分层附加可视对象,然后在它们无用时分离和销毁。

我认为您可以检查OnVisualParentChanged公开的方法UIElement:在附加视觉对象和分离视觉对象时调用此方法。这可能是处理非托管对象(套接字、文件等)的正确位置。

于 2009-10-11T16:05:35.910 回答
8

我也在寻找这个,在测试了不同的选项后,我实施了 venezia 的解决方案

protected override void OnVisualParentChanged(DependencyObject oldParent)
    {
        if (oldParent != null)
        {
            MyOwnDisposeMethod(); //Release all resources here
        }

        base.OnVisualParentChanged(oldParent);
    }

我意识到当父调用Children.Clear()Method 并且已经将项目添加到 Children 时,DependencyObject 有一个值。但是当父母添加了一个项目(Children.Add(CustomControl))并且孩子为空时,DependencyObject 为空。

于 2012-12-28T18:36:50.880 回答
0

尽管其他人已经为您提供了有关此问题的非常有用的信息,但您可能没有一些信息可以解释为什么没有 IDisposable。基本上,WPF(和 Silverlight)大量使用 Wea​​kReferences - 这允许您引用 GC 仍然可以收集的对象。

于 2009-10-11T20:53:41.600 回答
0

当我使用一些与数据库的连接,使用一些 IDbConnection 实现驱动程序或实体框架时,我遇到了这个困难。

我发现在这些情况下,建议是为每个窗口保留一个对象连接/上下文,以便能够跟踪更改/事务。(链接

所以我已经覆盖了 OnClosing:

protected override void OnClosing(System.ComponentModel.CancelEventArgs e)
{
      base.OnClosing(e);
      this._context.Dispose();
}

上下文可以是我的 ViewModel/Control,它实现了 IDisposable 来清理资源。

使用 OnClosing 处理资源的示例(链接

于 2020-10-28T14:31:56.527 回答