CSh*_*ark 3 c# wpf winapi .net-5
我目前正在尝试让 Global LowLevel 键盘挂钩在 .net 5 中使用 wpf 工作,但一段时间后我总是遇到一个问题,ExecutionEngineException并且我的应用程序崩溃了。根据文档,这种异常根本不应该再发生,但对我来说它确实发生了。它也不是特定于我的项目的,因为最小的 WPF .net 5 项目也会在一些按键混搭后崩溃。
有谁知道如何解决或解决这个问题?
主窗口的示例代码:
public partial class MainWindow : Window {
private const int WH_KEYBOARD_LL = 13;
private const int WM_KEYDOWN = 0x0100;
private const int WM_SYSKEYDOWN = 0x0104;
private delegate IntPtr LowLevelKeyboardProc(int nCode, IntPtr wParam, IntPtr lParam);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr SetWindowsHookEx(int idHook, LowLevelKeyboardProc lpfn, IntPtr hMod,
uint dwThreadId);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode,
IntPtr wParam, IntPtr lParam);
[DllImport("kernel32", SetLastError = true, CharSet = CharSet.Ansi)]
static extern IntPtr LoadLibrary([MarshalAs(UnmanagedType.LPStr)] string lpFileName);
private IntPtr _hook;
public bool Hook() {
if (_hook == IntPtr.Zero) {
var modPtr = LoadLibrary("user32.dll");
_hook = SetWindowsHookEx(WH_KEYBOARD_LL, Handler, modPtr, 0);
}
return _hook != IntPtr.Zero;
}
private IntPtr Handler(int code, IntPtr param, IntPtr lParam) {
return CallNextHookEx(_hook, code, param, lParam);
}
public MainWindow() {
InitializeComponent();
Hook();
}
}
Run Code Online (Sandbox Code Playgroud)
我得到的调用堆栈:
WindowsBase.dll!System.Windows.Threading.Dispatcher.GetMessage(ref System.Windows.Interop.MSG msg, System.IntPtr hwnd, int minMessage, int maxMessage)
WindowsBase.dll!System.Windows.Threading.Dispatcher.PushFrameImpl(System.Windows.Threading.DispatcherFrame frame)
WindowsBase.dll!System.Windows.Threading.Dispatcher.PushFrame(System.Windows.Threading.DispatcherFrame frame)
WindowsBase.dll!System.Windows.Threading.Dispatcher.Run()
PresentationFramework.dll!System.Windows.Application.RunDispatcher(object ignore)
PresentationFramework.dll!System.Windows.Application.RunInternal(System.Windows.Window window)
PresentationFramework.dll!System.Windows.Application.Run()
EngineExecutionError.dll!EngineExecutionError.App.Main()
Run Code Online (Sandbox Code Playgroud)
正如评论所指出的,您不应将 lambda 或委托直接传递到 Win32 Api 调用(或任何与此相关的 C api 调用)。修复方法是创建一个包含函数引用的静态变量
private static LowLevelKeyboardProc _handler = Handler;
[...]
_hook = SetWindowsHookEx(WH_KEYBOARD_LL, _handler, modPtr, 0);
Run Code Online (Sandbox Code Playgroud)
或者固定委托,这样 CLR 就不会移动它,这就是我的选择。
private LowLevelKeyboardProc _handler;
private GCHandle _gcHandler;
public KeyboardHook() {
_handler = Handler;
_gcHandler = GCHandle.Alloc(_handler);
}
[...]
_hook = SetWindowsHookEx(WH_KEYBOARD_LL, _handler, modPtr, 0);
Run Code Online (Sandbox Code Playgroud)
对于低级挂钩,您似乎可以设置modPtr为LoadLibrary("user32.dll"),GetModuleHandle(null)或IntPtr.Zero。似乎与他们所有人一起工作。
| 归档时间: |
|
| 查看次数: |
700 次 |
| 最近记录: |