11

我想在 C# 中获取当前窗口句柄数和系统范围的窗口句柄限制。我该怎么做?

4

3 回答 3

19

如果您阅读 Raymond Chen 的帖子,您可能会发现它和我一样烦人。您只是“可能做错了什么”,因为您正在做 Windows 无法做到的事情。

在我的应用程序中,当用户第一次访问标签页时,我创建并布置了该页上的所有控件。这需要相当长的时间——一个页面上很容易有 50 个控件。因此,如果可能的话,我不会在填充标签页后丢弃它,并将关闭的标签页集留给用户。

碰巧的是,一些用户从不想关闭任何一组标签页。我为什么要强迫他们这样做?使用我的 UI,他们可以非常快速地导航到他们负责管理的 300 多组交易中的任何一组。他们的机器足够快,并且有足够的内存,使这一切都非常敏感。唯一的问题是 Windows 不支持它。

为什么我使用控件,而不是其他一些 UI 技术?因为他们工作。我需要支持焦点事件、选项卡顺序、验证事件、动态布局和数据绑定——用户实际上是在内存数据集中管理数十个表中的数千条记录。我必须做的开发量 - 比如说 - 使用无窗口控件实现某些东西是天文数字。

我只是“做错了”,因为 Windows 对它可以支持的窗口句柄的数量有硬性限制。这个硬性限制是基于一堆关于如何构建计算机 UI 的十年前的假设。“做错事”的不是我。

无论如何,我对此的解决方案分为两部分。

首先,一个可以告诉您进程正在使用多少个窗口句柄的类:

using System;
using System.Runtime.InteropServices;

namespace StreamWrite.Proceedings.Client
{
    public class HWndCounter
    {
        [DllImport("kernel32.dll")]
        private static extern IntPtr GetCurrentProcess();

        [DllImport("user32.dll")]
        private static extern uint GetGuiResources(IntPtr hProcess, uint uiFlags);

        private enum ResourceType
        {
            Gdi = 0,
            User = 1
        }

        public static int GetWindowHandlesForCurrentProcess(IntPtr hWnd)
        {
            IntPtr processHandle = GetCurrentProcess();
            uint gdiObjects = GetGuiResources(processHandle, (uint)ResourceType.Gdi);
            uint userObjects = GetGuiResources(processHandle, (uint)ResourceType.User);

            return Convert.ToInt32(gdiObjects + userObjects);
        }
    }
}

其次,我维护一个最近最少使用的标签页对象缓存。.NET 框架不提供通用 LRU 缓存类,因此我构建了一个,如果需要,可以在此处获取。每次用户访问标签页时,我都会将其添加到 LRU 缓存中。然后我检查我是否在窗口句柄上运行不足。如果是,我会丢弃最近最少使用的标签页上的控件,并继续这样做,直到我再次有足够的窗口句柄为止。

于 2009-02-11T00:50:24.423 回答
3

正如 Raymond Chen 前段时间所说,如果您正在考虑窗口句柄限制,那么您可能做错了什么:)

无论如何,我敢打赌没有特殊的 C# 方法可以做到这一点,因为它是非常特定于系统的。您可以使用在 C++ 应用程序中使用的相同函数。使用 P/Invoke 调用函数。要了解如何编写导入,请访问pinvoke.net

编辑:据我了解您的问题,我假设您已经知道如何在 Win32 应用程序中执行此操作。

于 2008-09-27T08:17:05.847 回答
0

OregonGhost 引用的完整报价是

如果你不得不问,你可能做错了什么。

它来自为什么每个进程的窗口句柄限制为 10,000?你应该读这个。

于 2008-09-27T09:31:10.443 回答