我在C#代码中使用Windows CBT钩子,如下所示。这段代码效果很好,但我刚想到我对它的实际工作原理并不清楚。
那么,从非托管Windows代码到常规函数指针(CBTProc)的回调如何变成非静态托管成员方法调用,因此我可以访问this
中的CbtHookProc
}?
通常,我了解C#委托如何捕获托管代码中的范围。我的问题是它如何仍然适用于非托管回调。在本机C ++中,如果不使用全局变量或某些技巧(如ATL thunk),这是不可能的。
class CbtHook
{
delegate IntPtr HookProc(int nCode, IntPtr wParam, IntPtr lParam);
HookProc cbtHookProc;
IntPtr cbtHookId;
const int WH_CBT = 5;
[DllImport("user32.dll")]
static extern IntPtr SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hInstance, uint threadId);
[DllImport("kernel32.dll")]
static extern uint GetCurrentThreadId();
[DllImport("user32.dll")]
static extern IntPtr CallNextHookEx(IntPtr hhook, int nCode, IntPtr wParam, IntPtr lParam);
IntPtr CbtHookProc(int nCode, IntPtr wParam, IntPtr lParam)
{
// can access 'this' here
return CallNextHookEx(this.cbtHookId, nCode, wParam, lParam);
}
public CbtHook()
{
this.cbtHookProc = new HookProc(CbtHookProc);
this.cbtHookId = SetWindowsHookEx(WH_CBT, this.cbtHookProc, IntPtr.Zero, GetCurrentThreadId());
}
}
答案 0 :(得分:2)
来自MSDN:
当委托被编组为函数指针时,运行时会分配一个执行从非托管到托管的转换的thunk。这个thunk是非托管代码在最终调用托管委托之前实际调用的内容。