SetWindowsHookEx失败,错误126

我正在尝试在项目中使用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并且上面的代码不断抛出带有消息的exceptionThe specified module could not be found并且对Marshal.GetLastWin32Error的调用返回代码126 。 我可以成功运行Gma.UserActivityMonitor原始项目提供的演示,但由于我的项目有点太复杂而无法解释,我无法详细解释它与我的不同之处。 我只是希望有人能盲目地猜出这个问题。

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

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

这种代码在.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(),该模块将保持加载状态,直到程序终止。