19

我正在尝试在项目中使用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}和我的项目中抛出异常的相同!

4

1 回答 1

34

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

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

SetWindowsHookEx() 对于低级钩子来说有点尴尬。它需要一个有效的模块句柄,并检查它,但实际上并不使用它。这在 Windows 中得到了修复,在 Win7 SP1 附近的某个地方。虽然肯定打算成为一个有用的修复程序,但它实际上使问题变得更糟。因为现在它可以在您的开发机器上运行,但不能在您用户的机器上运行。

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

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

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

于 2013-07-27T13:07:54.627 回答