SetWindowsHookEx失败,返回错误126

本文关键字:错误 返回 失败 SetWindowsHookEx | 更新日期: 2023-09-27 18:25:13

我正试图在一个项目中使用Gma.UserActivityMonitor库,但我遇到了一个无法独自克服的错误。

HookManager.Callbacks.cs文件中,有一个名为EnsureSubscribedToGlobalMouseEvents的静态方法,其代码(或多或少)如下:

var asm = Assembly.GetExecutingAssembly().GetModules()[0];
var mar = Marshal.GetHINSTANCE(asm);
s_MouseHookHandle = SetWindowsHookEx(
    WH_MOUSE_LL,
    s_MouseDelegate,
    mar,
    0);
//If SetWindowsHookEx fails.
if (s_MouseHookHandle == 0)
{
    //Returns the error code returned by the last unmanaged function called using platform invoke that has the DllImportAttribute.SetLastError flag set. 
    int errorCode = Marshal.GetLastWin32Error();
    //do cleanup
    //Initializes and throws a new instance of the Win32Exception class with the specified error. 
    throw new Win32Exception(errorCode);
}

SetWindowsHookEx总是返回0,上面的代码不断抛出异常消息The specified module could not be found,对Marshal.GetLastWin32Error的调用返回代码126。我可以成功地运行Gma.UserActivityMonitor的原始项目提供的演示,但由于我的项目有点太复杂,无法在这里解释,我无法详细解释它与我的项目的区别。我只是希望有人能盲猜出这个问题。

顺便说一句,在该项目的常见问题解答中,据说当仅在调试项目时检查Enable Visual Studio hosting process时,其他人有一个接近我的问题(SetWindowsHookEx返回错误)。因此,我取消了我的复选框,但我仍然有同样的问题,不仅在调试模式下,而且当我在Windows资源管理器中双击发布文件时(不涉及Visual Studio)。

为了提供更多信息,在演示项目中(工作正常),asm变量指向{Gma.UserActivityMonitor.dll},在我的项目中也是如此,抛出了异常!

SetWindowsHookEx失败,返回错误126

这种代码在.NET4及更高版本上不再工作。您得到的错误代码在其他方面是描述性的,126="找不到指定的模块"。这告诉你"mar"变量包含垃圾。

.NET4对CLR进行了相当重大的更改,它不再假装jitted代码位于非托管模块中。所以Marshal.GetHINSTANCE()不再工作。然后代码变得草率,忘记检查返回值,需要测试(IntPtr)-1来检测故障并宣布灾难。对于您在Codeproject中发现的代码来说,非常常见的是,有很多错误和草率是贡献者无法修复的。不是SO型号:)

SetWindowsHookEx()对于低级钩子来说有点尴尬。它需要一个有效的模块句柄,并检查它,但实际上并没有使用它。这在Windows中得到了修复,大约在Win7 SP1中。虽然这当然是一个有用的解决方案,但实际上却让问题变得更糟。因为现在它可以在您的开发机器上工作,但不能在用户的机器上工作。

总之,修复很简单,你只需要拿出一个有效的模块句柄。你可以从托管应用程序中始终存在的模块中获得一个,你需要pinvoke LoadLibrary才能获得它:

var mar = LoadLibrary("user32.dll");
s_MouseHookHandle = SetWindowsHookEx(
    WH_MOUSE_LL,
    s_MouseDelegate,
    mar,
    0);

无需调用FreeLibrary(),该模块将保持加载状态,直到程序终止。