【问题标题】:Windows Form Memory leakWindows 窗体内存泄漏
【发布时间】:2014-08-07 11:45:20
【问题描述】:

我在我的 Windows 应用程序中看到轻微的内存泄漏。我在我的应用程序中使用 DevExpress XtraForm。我看到的是表单的一个实例始终保存在内存中。如果您多次打开同一个表单,它仍然会保留上次打开的表单的引用。

例如。如果您在应用程序中打开 10 个不同的表单并关闭所有表单,由于一些奇怪的“MdiClient 对象引用 LayoutEventArgs 对象”,它仍然不会释放分配给它的内存。幸运的是,它保留了每种类型的单个项目的引用。

这里是 Redgate 内存分析器输出的链接。

https://dl.dropboxusercontent.com/u/2781659/Memory%20Leak.pdf

在上面的图表中,DepartmentsForm 已被处理,但由于 LayoutEventArgs 的受影响组件成员引用它而无法被 GC。

如果您发现任何明显的错误,请告知。

【问题讨论】:

  • 没有任何代码就无法判断,即使这样我也怀疑它可能很难提供帮助
  • 我已经提供了对象保留图以及问题。

标签: c# .net memory-leaks devexpress devexpress-windows-ui


【解决方案1】:

根据我的经验,在 Windows 窗体中存在一些情况,当处置的控件可以缓存在 LayoutEventArgs 对象中时,它看起来像是 WinForms 中的某种小错误。

一些细节:
System.Windows.Forms.Control 类型的每个实例都包含一个 LayoutEventArgstype - cachedLayoutEventArgs 的私有成员变量。而且,LayoutEventArgs 通常包含对某些特定控件的引用。您可以通过 Reflector 清楚地看到所有这些事实。并且,有时,由于某些原因,当子控件的处置不影响父控件的布局过程时,cachedLayoutEventArgs字段没有被清除。您可以使用 mdi 父窗体来模拟这种情况,方法是在关闭其子级时暂停 MdiClient 的控件布局:

public partial class MdiParentForm : Form {
    public MdiParentForm () {
        InitializeComponent(); //  this.IsMdiContainer = true
    }
    void buttonAddMdiChild_Click(object sender, EventArgs e) {
        MdiChildForm f = new MdiChildForm();
        f.MdiParent = this;
        f.Show();
    }
    void buttonCloseMdiChild_Click(object sender, EventArgs e) {
        MdiClient client = GetMdiClient(this);
        client.SuspendLayout();

        if(ActiveMdiChild != null)
            ActiveMdiChild.Close();

        client.ResumeLayout(false); 
        // !!! At this point the MdiClient.cachedLayoutEventArgs contains the reference to disposed control (leak)
    }
    static MdiClient GetMdiClient(Form frm) {
        if(frm != null) {
            foreach(Control ctrl in frm.Controls) {
                if(ctrl is MdiClient)
                    return (MdiClient)ctrl;
            }
        }
        return null;
    }
}
class MdiChildForm : Form { }

有一个简单的解决方法 - 通过触发 PerformLayout 方法,您可以有效地清除“缓存”实例:

class MdiChildForm : Form {
    MdiClient parent;
    protected override void OnParentChanged(EventArgs e) {
        base.OnParentChanged(e);
        var mdiClient = Parent as MdiClient;
        if(mdiClient != parent) {
            if(parent != null)
                parent.PerformLayout();
            parent = mdiClient;
        }
    }
}

附:无论如何我建议您联系DevExpress support这方面,以确保您描述的内存泄漏与他们的控件无关并得到最终的解决方案。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2011-04-22
    • 1970-01-01
    • 1970-01-01
    • 2020-04-02
    • 2014-05-03
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多