4

我目前正在调试一个存在内存泄漏问题的大型 Winforms 应用程序。我使用 .NET 内存分析器,到目前为止,我已经能够找到其中的一些泄漏并解决它们。但是现在我面临一个问题,我不确定是不是问题,如果是问题,我不知道如何解决。

在运行我的应用程序大约 1 分钟后(考虑到普通用户可以使用几个小时,这并不算多),.NET 内存分析器向我显示了来自 Krypton Toolkit 的不同控件的大约 100-200 个实例,如果我这样做,这个数字会增加继续前进(它们永远不会被垃圾收集,因为看起来它们仍在某处被引用)。现在,如果我检查这些实例的根路径,它们看起来都像:

Krypton ViewManager 的根路径

我不知道在我的代码中哪里可以在不再需要这些实例时正确地取消引用它们,因为我不知道什么仍在引用控件。我知道创建 KryptonButtonEx 的位置,据我所知,ViewManager 是由这个按钮创建的,但我仍然不知道我能做些什么。对于那些感兴趣的人,创建按钮的代码是这样的:

        KryptonButton newControlButton = new KryptonButton();
        newControlButton.Tag = mtActivityControl;
        newControlButton.Anchor = ((System.Windows.Forms.AnchorStyles)((System.Windows.Forms.AnchorStyles.Left | System.Windows.Forms.AnchorStyles.Right)));
        newControlButton.AutoSize = true;
        newControlButton.AutoSizeMode = System.Windows.Forms.AutoSizeMode.GrowOnly;
        newControlButton.ButtonStyle = ComponentFactory.Krypton.Toolkit.ButtonStyle.ListItem;
        newControlButton.Location = new System.Drawing.Point(3, 3);
        newControlButton.Name = string.Format("controlButton{0}", mtActivityControl.SymbolicName);
        newControlButton.Size = new System.Drawing.Size(96, 23);
        newControlButton.StateCommon.Content.Image.ImageH = ComponentFactory.Krypton.Toolkit.PaletteRelativeAlign.Near;
        newControlButton.StateCommon.Content.ShortText.TextH = ComponentFactory.Krypton.Toolkit.PaletteRelativeAlign.Near;
        newControlButton.TabIndex = 5;

        StringBuilder buttonText = new StringBuilder();
        buttonText.Append(Path.GetFileName(mtActivityControl.ControlName));
        /*if (mtActivityControl.SymbolicName.Length != 0)
        {
            buttonText.Append(" (");
            buttonText.Append(mtActivityControl.SymbolicName);
            buttonText.Append(")");
        }*/

        newControlButton.Text = buttonText.ToString();
        newControlButton.Values.ExtraText = "";
        newControlButton.Values.Image = null;
        newControlButton.Values.ImageStates.ImageCheckedNormal = null;
        newControlButton.Values.ImageStates.ImageCheckedPressed = null;
        newControlButton.Values.ImageStates.ImageCheckedTracking = null;
        newControlButton.Values.Text = buttonText.ToString();
        newControlButton.Click += new System.EventHandler(this.controlsButton_Click);

尽管我的研究告诉我没有必要,我还是在 Dispose 函数中取消了这样的事件:

newControlButton.Click -= new System.EventHandler(this.controlsButton_Click);

所以我的问题是:

Krypton 本身是否有可能保留对我的控件的引用,这导致一些内存没有被释放(如果它是用于保留对象池或类似东西的有限内存量,这可能是可以的,但如果这是一个不受控制的内存泄漏)?如果它不是来自 Krypton,您是否知道在哪里可以正确销毁这些实例?

非常感谢!

编辑:

我刚刚注意到 KryptonButtonEx 类不是来自 Krypton,而是来自我的应用程序。但我认为它不会改变任何问题,因为它唯一要做的就是覆盖 GetPreferredSize 函数:

/// <summary>
/// An extended/fixed KryptonButton which handles resizing correctly.
/// </summary>
public class KryptonButtonEx : ComponentFactory.Krypton.Toolkit.KryptonButton
{
    /// <summary>
    /// Gets the size of the preferred.
    /// </summary>
    /// <param name="proposedSize">Size of the proposed.</param>
    /// <returns></returns>
    public override Size GetPreferredSize(Size proposedSize)
    {
        // Do we have a manager to ask for a preferred size?
        if (ViewManager != null)
        {
            // Ask the view to peform a layout
            Size retSize = ViewManager.GetPreferredSize(Renderer, proposedSize);

            // Apply the maximum sizing
            if (MaximumSize.Width > 0) retSize.Width = Math.Min(MaximumSize.Width, retSize.Width);
            if (MaximumSize.Height > 0) retSize.Height = Math.Min(MaximumSize.Height, retSize.Width);

            // Apply the minimum sizing
            if (MinimumSize.Width > 0) retSize.Width = Math.Max(MinimumSize.Width, retSize.Width);
            if (MinimumSize.Height > 0) retSize.Height = Math.Max(MinimumSize.Height, retSize.Height);

            return retSize;
        }
        else
        {
            // Fall back on default control processing
            return base.GetPreferredSize(proposedSize);
        }
    }
}
4

1 回答 1

5

您发布了错误的代码。您应该对删除它的代码非常感兴趣,而不是创建按钮的代码。搜索 Controls.Remove 和 Controls.Clear 并确保正在释放每个要删除的控件。另一个诊断是 TaskMgr.exe,进程选项卡。查看 + 选择列并勾选 USER 对象。看到这个数字稳步上升是一个硬线索,表明代码没有在应该的地方配置控件。这是我知道的唯一不调用 Dispose() 导致永久泄漏的情况。

使用如下代码代替 Controls.Clear():

 while (panel.Controls.Count > 0) panel.Controls[0].Dispose();

释放控件还会自动将其从父控件的集合中移除。

于 2011-05-27T13:08:45.703 回答