SetWindowsHookEx失败,错误126

时间:2013-07-27 12:09:01

标签: c# setwindowshookex

我正在尝试在项目中使用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原始项目提供的演示,但由于我的项目有点太复杂而无法在此解释,我无法详细解释它与我的不同之处。我只是希望有人能盲目地猜出这个问题。

顺便说一句,在项目的常见问题解答中,当仅在调试项目时检查SetWindowsHookEx时,其他人在我的附近遇到问题(Enable Visual Studio hosting process返回错误)。所以我取消选中我的那个盒子,但我仍然遇到同样的问题,不仅仅是在调试模式下,而且当我在Windows资源管理器中双击发布文件时(不涉及Visual Studio)。

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

1 个答案:

答案 0 :(得分:28)

这种代码在.NET 4及更高版本上不再起作用。您获得的错误代码是描述性的,126 =“无法找到指定的模块”。这告诉你“mar”变量包含垃圾。

.NET 4有一个非常重要的CLR更改,它不再假装jitted代码存在于非托管模块中。所以Marshal.GetHINSTANCE()不再起作用了。然后代码变得草率,忘记检查返回值,测试它(IntPtr)-1是检测失败和声明灾难所必需的。您在Codeproject上找到的代码很常见,很多错误和邋is都无法由贡献者修复。不是SO型号:)

SetWindowsHookEx()对于低级挂钩有点尴尬。它需要一个有效的模块句柄,并检查它,但实际上并没有使用它。这已在Windows中修复,在Win7 SP1附近。虽然肯定是一个有用的解决方案,但它实际上使问题变得更糟。因为现在它可以在您的开发机器上运行,但不能在用户的机器上运行。

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

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

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