2

每当用户按下窗口中的某些键时,我的应用程序应该执行一些操作。

SetWindowsHookEx使用选项调用WH_KEYBOARD_LL似乎是实现这一目标的标准方法。但是,在我的情况下,某些事情显然是错误的,并且没有触发回调。

我的调试控制台应用程序的主要方法:

static void Main(string[] args)
{
    IntPtr moduleHandle = GetCurrentModuleHandle();
    IntPtr hookHandle = IntPtr.Zero;

    try
    {
        User32.HookProc hook = (nCode, wParam, lParam) =>
        {
            // code is never called :-(
            if (nCode >= 0)
            {
                Console.WriteLine("{0}, {1}", wParam.ToInt32(), lParam.ToInt32());
            }
            return User32.CallNextHookEx(hookHandle, nCode, wParam, lParam);
        };

        hookHandle = User32.SetWindowsHookEx(User32.WH_KEYBOARD_LL, hook, moduleHandle, 0);

        Console.ReadLine(); // 
    }
    finally
    {
        if (hoodHandle != IntPtr.Zero)
        {
            var unhooked = User32.UnhookWindowsHookEx(hookHandle);
            Console.WriteLine(unhooked); // true
            hookHandle = IntPtr.Zero;                   
        }
    }
}

GetCurrentModuleHandle 方法:

private static IntPtr GetCurrentModuleHandle()
{
    using (var currentProcess = Process.GetCurrentProcess())
    using (var mainModule = currentProcess.MainModule)
    {
        var moduleName = mainModule.ModuleName;
        return Kernel32.GetModuleHandle(moduleName);
    }           
}

user32.dll从和进口kernel32.dll

public static class User32
{
    public const int WH_KEYBOARD_LL = 13;

    public delegate IntPtr HookProc(int nCode, IntPtr wParam, IntPtr lParam);

    [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    public static extern IntPtr SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hMod, uint dwThreadId);

    [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    [return: MarshalAs(UnmanagedType.Bool)]
    public static extern bool UnhookWindowsHookEx(IntPtr hhk);

    [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    public static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);        
}

public static class Kernel32
{
    [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    public static extern IntPtr GetModuleHandle(string lpModuleName);
}

你知道我是什么问题吗?

4

3 回答 3

4

无法从 C# 控制台应用程序调用 SetwindowHook API。您需要在 Windows DLL 中调用它(它在 .Net DLL 中不起作用)。

当然有一种方法可以解决这个问题,而且我确实使用它作为我很久以前构建的应用程序之一。但我现在不记得了。如果您搜索的时间足够长,您可能会发现。

于 2010-10-21T17:02:21.357 回答
3

这是 Main(string[] args) 和 Console.ReadLine() 建议的控制台应用程序吗?

如果是这样,那么这可能是您的问题的根源

于 2010-10-21T18:21:41.793 回答
2

当您像这样使用 Win32 API 时,您自己检查错误非常重要。您不再拥有可以为您执行此操作并引发异常的友好 .NET 包装器。这需要在几个地方完成,但这里有一个:

    hookHandle = User32.SetWindowsHookEx(User32.WH_KEYBOARD_LL, hook, moduleHandle, 0);
    if (hookHandle == IntPtr.Zero) throw new Win32Exception();

真正的问题是您对 mainModule.ModuleName 的使用。如果只给出文件名,而不是完整路径。GetModuleHandle() 将失败并返回 IntPtr.Zero。如上所述,如果您对此进行了测试,那么您很快就会发现问题。

当您在 .NET 4.0 中运行此代码时,还有一个额外的故障模式。CLR 不再为托管代码伪造本机模块。SetWindowsHookEx() 需要一个有效的 DLL 句柄,但它实际上并没有使用它,因为这是一个低级挂钩。获得一个的最好方法是请求 user32.dll,它总是加载在托管程序中:

    IntPtr moduleHandle = LoadLibrary("user32.dll");
    if (moduleHandle == IntPtr.Zero) throw new Win32Exception();

不需要释放它。

于 2010-10-21T17:21:50.470 回答