4

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

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

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

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

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

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

4

1 回答 1

9

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

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

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;
        }
    }
}

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

于 2014-08-07T17:47:32.863 回答