我正在尝试在项目中使用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}
,并在我的项目中引发异常!
答案 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(),该模块会一直保持加载状态,直到您的程序终止。