这几天用C#做了一个抓图软件,其中用到了系统钩子技术。

在这里发表下自己用C#实现系统钩子的心得,希望对我的朋友们有所帮助。

看了很多资料,都提到系统钩子(Hook)必须是一个dll,这样Windows才能在所有进程中共享钩子代码。

与钩子相关的3个Windows API函数是:

SetWindowsHookEx、UnhookWindowsHookEx、CallNextHookEx,它们的简要说明如下:

C#系统钩子(转载)HHOOK SetWindowsHookEx( 
C#系统钩子(转载)
int  idHook, 
C#系统钩子(转载)HOOKPROC lpfn, 
C#系统钩子(转载)HINSTANCE hMod, 
C#系统钩子(转载)DWORD dwThreadId); 
C#系统钩子(转载)

 

idHook -- 系统钩子类型,实际上是一个int值,具体的定义请看MSDN

lpfn -- 钩子回调函数指针,当收到钩子消息时就执行这个回调函数,回调函数有3个参数,不同类型的钩子这3个参数有不同的含义,具体请看MSDN。

hMod -- DLL实例句柄(我猜是这样,我不熟悉Win32API编程),当钩子为系统钩子时这个参数是必须的,如果是单个程序或者线程使用的钩子那么可以不用这个值。

dwThreadId -- 线程ID,如果是单个程序或者线程使用的钩子这个参数是必须的。

SetWindowsHookEx返回所设置的钩子的句柄。

BOOL UnhookWindowsHookEx(HHOOK hhk);

hhk -- 所要注销的钩子的句柄

UnhookWindowsHookEx返回注销钩子是否成功。

C#系统钩子(转载)LRESULT CallNextHookEx( 
C#系统钩子(转载)HHOOK hhk, 
C#系统钩子(转载)
int  nCode, 
C#系统钩子(转载)WPARAM wParam, 
C#系统钩子(转载)LPARAM lParam); 

hhk -- 起始钩子句柄,如果是在钩子回调函数中调用CallNextHookEx时传的通常是本钩子句柄。

nCode -- 钩子回调函数参数1,具体含义视钩子类型而定。

wParam -- 钩子回调函数参数2,具体含义视钩子类型而定。

lParam -- 钩子回调函数参数3,具体含义视钩子类型而定。

从以上3个函数的简要说明可以看出所用到的参数几乎都是简单的参数,稍微复杂就是回调函数(在C#中用delegate类型来实现,后面会提到),但是最让人头疼的就是DLL的实例句柄,C#制作的DLL并不像C做的DLL有DllMain入口可以接受到DLL实例句柄。

我目前唯一的办法就是使用C做一个系统钩子DLL再让C#调用。底下是DLL的代码。

C#系统钩子(转载)#define  _WIN32_WINNT 0x0500 
C#系统钩子(转载)
C#系统钩子(转载)#include 
< Windows.h >  
C#系统钩子(转载)#include 
< WinUser.h >  
C#系统钩子(转载)
C#系统钩子(转载)
#define  Dll_Export __declspec(dllexport) 
C#系统钩子(转载)
C#系统钩子(转载)public   class  Hook 


 上面用到的User32.HookProc类型不是.NET框架提供的,是自己定义的一个委托类型,声明如下

C#系统钩子(转载)public   delegate   int  HookProc( int  nCode, IntPtr wParam, IntPtr lParam); 

现在你的C#程序就可以使用各种系统钩子了,我们目前已经使用的有低级别鼠标钩子和低级别键盘钩子。

应用低级别鼠标钩子时会遇到这样一个问题:

低级别鼠标钩子的回调函数lParam是一个MSLLHOOKSTRUCT结构体指针,C#中怎么获取这个结构体的实例呢?实现代码如下

C#系统钩子(转载)MSLLHOOKSTRUCT _mstr; 
C#系统钩子(转载)
unsafe
 }

通过声明非安全代码来获取指针所指对象。

MSLLHOOKSTRUCT结构体的C#定义如下:

C#系统钩子(转载)[StructLayout(LayoutKind.Sequential)] 
C#系统钩子(转载)
public   struct  MSLLHOOKSTRUCT 
 }

      PS: 附件--系统钩子的源代码与DLL

补充HookType枚举值,很抱歉一时疏忽忘记写出来,这些值可以在Win32 API 的相关头文件中找到,有LL字样的就是低级钩子:

C#系统钩子(转载)
C#系统钩子(转载)        
public   enum  HookTypes:  int
        }


虽然上篇文章已经有但是还是要重点重复说明下CallNextHookEx函数,这个函数的功能是在执行完系统钩子的回调函数后接着调用下一个钩子函数,当设置的钩子是底层键盘或者鼠标钩子时如果不这么做将会时键盘和鼠标"失灵"。

C#系统钩子(转载)        [DllImport("user32.dll")]
C#系统钩子(转载)        
public static extern int CallNextHookEx(IntPtr hookHandle, int nCode, IntPtr wParam, IntPtr lParam);

这个函数第一个参数是钩子的句柄,在创建钩子时返回来的那个就是,后面3个参数和钩子回调函数的3个参数一一对应,照传就可以。

因为公司现在装得是VS2005所以我只能先用VS2005做个试验程序了,这个程序功能是锁定鼠标一定时间,需要有.NET Framework 2.0 Runtime才能运行,我会尽快找台装VS2003的电脑做个.NET 1.1的范例的。

可执行文件:https://files.cnblogs.com/BG5SBK/HookTest_NET2_Bin.rar
项目文件:https://files.cnblogs.com/BG5SBK/HookTest_NET2_Source.rar
系统钩子文件:https://files.cnblogs.com/BG5SBK/LowLevelHook.rar

 

 

原文地址:http://www.cnblogs.com/bg5sbk/articles/299728.html

相关文章:

  • 2022-12-23
  • 2021-11-11
  • 2021-03-31
  • 2021-11-24
  • 2021-05-03
  • 2022-12-23
  • 2021-08-16
  • 2022-01-05
猜你喜欢
  • 2021-10-27
  • 2022-12-23
  • 2021-09-04
  • 2022-02-07
  • 2022-12-23
  • 2022-01-18
  • 2022-02-13
相关资源
相似解决方案