這里就以上文提到的簡單例子做個線程鉤子。
第一步:聲明API函數
// 安裝鉤子
[DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
public static extern int SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hInstance, int threadId);
// 卸載鉤子
[DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
public static extern bool UnhookWindowsHookEx(int idHook);
// 繼續下一個鉤子
[DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
public static extern int CallNextHookEx(int idHook, int nCode, Int32 wParam, IntPtr lParam);
// 取得當前線程編號
[DllImport("kernel32.dll")]
static extern int GetCurrentThreadId();
#endregion
聲明一下API函數,以后就可以直接調用了。
第二步:聲明、定義。
static int hKeyboardHook = 0;//如果hKeyboardHook不為0則說明鉤子安裝成功
HookProc KeyboardHookProcedure;
#endregion
先解釋一下委托,鉤子必須使用標準的鉤子子程,鉤子子程就是一段方法,就是處理上面例子中提到的讓TextBox顯示“A”的操作。
鉤子子程必須按照HookProc(int nCode, Int32 wParam, IntPtr lParam)這種結構定義,三個參數會得到關于消息的數據。
當使用SetWindowsHookEx函數安裝鉤子成功后會返回鉤子子程的句柄,hKeyboardHook變量記錄返回的句柄,如果hKeyboardHook不為0則說明鉤子安裝成功。
第三步:寫鉤子子程
我們寫一個方法,返回一個int值,包括三個參數。如上面給出的代碼,符合鉤子子程的標準。
nCode參數是鉤子代碼,鉤子子程使用這個參數來確定任務,這個參數的值依賴于Hook類型。
wParam和lParam參數包含了消息信息,我們可以從中提取需要的信息。
方法的內容可以根據需要編寫,我們需要TextBox顯示“ fangqm.cn”,那我們就寫在這里。當鉤子截獲到消息后就會調用鉤子子程,這段程序結束后才往下進行。截獲的消息怎么處理就要看子程的返回值了,如果返回1,則結束消息,這個消息到此為止,不再傳遞。如果返回0或調用CallNextHookEx函數則消息出了這個鉤子繼續往下傳遞,也就是傳給消息真正的接受者。
第四步:正式啟用鉤子:安裝鉤子、卸載鉤子
準備工作都完成了,剩下的就是把鉤子裝入鉤子鏈表。
我們可以寫兩個方法在程序中合適位置調用。代碼如下:
if (hKeyboardHook == 0)
{
//終止鉤子
throw new Exception("安裝鉤子失敗");
}
}
}
//鉤子卸載
public void HookStop()
{
bool retKeyboard = true;
if (hKeyboardHook != 0)
{
retKeyboard = UnhookWindowsHookEx(hKeyboardHook);
hKeyboardHook = 0;
}
if (!retKeyboard)
throw new Exception("鉤子卸載失敗");
}
#endregion
安裝鉤子和卸載鉤子關鍵就是SetWindowsHookEx和UnhookWindowsHookEx方法。
SetWindowsHookEx (int idHook, HookProc lpfn, IntPtr hInstance, int threadId) 函數將鉤子加入到鉤子鏈表中,說明一下四個參數:
idHook 鉤子類型,即確定鉤子監聽何種消息,上面的代碼中設為2,即監聽鍵盤消息并且是線程鉤子,如果是全局鉤子監聽鍵盤消息應設為13,線程鉤子監聽鼠標消息設為7,全局鉤子監聽鼠標消息設為14。
lpfn 鉤子子程的地址指針。如果dwThreadId參數為0 或是一個由別的進程創建的線程的標識,lpfn必須指向DLL中的鉤子子程。 除此以外,lpfn可以指向當前進程的一段鉤子子程代碼。鉤子函數的入口地址,當鉤子鉤到任何消息后便調用這個函數。
hInstance應用程序實例的句柄。標識包含lpfn所指的子程的DLL。如果threadId 標識當前進程創建的一個線程,而且子程代碼位于當前進程,hInstance必須為NULL??梢院芎唵蔚脑O定其為本應用程序的實例句柄。
threaded 與安裝的鉤子子程相關聯的線程的標識符。如果為0,鉤子子程與所有的線程關聯,即為全局鉤子。
上面代碼中的SetWindowsHookEx方法安裝的是線程鉤子,用GetCurrentThreadId()函數得到當前的線程ID,鉤子就只監聽當前線程的鍵盤消息。
UnhookWindowsHookEx (int idHook) 函數用來卸載鉤子,卸載鉤子與加入鉤子鏈表的順序無關,并非后進先出。
四。安裝全局鉤子
上文使用的是線程鉤子,如果要使用全局鉤子在鉤子的安裝上略有不同。如下:
SetWindowsHookEx( 13,KeyboardHookProcedure,
Marshal.GetHINSTANCE(Assembly.GetExecutingAssembly().GetModules()[0]),0)
這條語句即定義全局鉤子。
子程消息處理
鉤子子程可以得到兩個關于消息信息的參數wPrama、lParam。怎么將這兩個參數轉成我們更容易理解的消息呢。
對于鼠標消息,我們可以定義下面這個結構:
對于鍵盤消息,我們可以定義下面這個結構:
然后我們可以在子程里用下面語句將lParam數據轉換成MSG或KeyMSG結構數據
MSG m = (MSG) Marshal.PtrToStructure(lParam, typeof(MSG));
KeyMSG m = (KeyMSG) Marshal.PtrToStructure(lParam, typeof(KeyMSG));
這樣可以更方便的得到鼠標消息或鍵盤消息的相關信息,例如p即為鼠標坐標,HWnd即為鼠標點擊的控件的句柄,vkCode即為按鍵代碼。
注:這條語句對于監聽鼠標消息的線程鉤子和全局鉤子都可以使用,但對監聽鍵盤消息的線程鉤子使用會出錯,目前在找原因。
如果是監聽鍵盤消息的線程鉤子,我們可以根據lParam值的正負確定按鍵是按下還是抬起,根據wParam值確定是按下哪個鍵。
如果是監聽鍵盤消息的全局鉤子,按鍵是按下還是抬起要根據wParam值確定。
wParam = = 0x100 // 鍵盤按下
wParam = = 0x101 // 鍵盤抬起
新聞熱點
疑難解答