a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁 > 編程 > Delphi > 正文

用DELPHI編制Windows95下的鉤子函數(shù)

2019-11-18 18:52:44
字體:
供稿:網(wǎng)友
Windows消息管理機構(gòu)提供了能使應(yīng)用程序訪問控制消息流μ
'c4所謂的鉤子(HOOK)機制。鉤子有多種,分別用于捕獲某一特定類型或某一范圍的消息。如:鍵盤消息,鼠標(biāo)消息等。我們這里僅以鍵盤鉤子的使用為例,討論在DELPHI下怎樣編寫DLL程序和怎樣在自己的程序中安裝使用鍵盤鉤子函數(shù),并討論了不同程序使用同一DLL文件時怎樣共享數(shù)據(jù)。
一、 鉤子過濾函數(shù)的編寫說明
由于鉤子過濾函數(shù)必須在獨立的模塊中,也就是說我們必須首先生成一個DLL框架,然后再在其中加入鉤子函數(shù)代碼以及其他相關(guān)函數(shù)代碼。我們這里以鍵盤鉤子過濾函數(shù)的編寫為例來說明。具體步驟如下:
1、先生成一個DLL筐2架
2、編寫自己的鍵盤鉤子過濾函數(shù)
鉤子過濾函數(shù)必須是回調(diào)函數(shù),其函數(shù)的 4?形如下?o
function KeyHookPRoc(
iCode:Integer;
wParam:WPARAM;
lParam:LPARAM ) : LRESULT; stdcall ;export ;
在生成的DLL框架中加入自己的鍵盤鉤子處理函數(shù)處理鍵盤消息。
代碼如下:…
if(iCode>=0) then begin
Result:=0; //初始化返回值
// 在這里加入自己的代碼
end else
begin
Result:=CallNextHook(hOldKeyHook,iCode,wParam,lParam);
// hOldKeyHook是保存的原鍵盤過濾函數(shù) 5刂?
end;
3、 安裝鍵盤鉤子過濾函數(shù)
為安裝一個鉤子?_fd濾函數(shù)應(yīng)調(diào)用SetWindowsHookEx函數(shù)(適用于Windows3.0的SetWindowsHook鉤子安裝函數(shù)現(xiàn)在已經(jīng)廢棄不用)。該函數(shù)的原形如下:
HHOOK SetWindowsHookEx(
int idHook,  // 安裝的?_b3子類型
HOOKPROC lpfn,  // 鉤子過濾??f數(shù)地址
HINSTANCE hMod,   // 任務(wù)句柄
DWord dwThreadId   // 鉤子用于的目的
);
需要說明的是:?_a8常應(yīng)該調(diào)用MakeProcInstance函數(shù)以獲取一個輸出函數(shù)的前導(dǎo)碼的入口地址,再將此地址作為SetWindowsHookEx的第二個參數(shù)lpfn。但由于Delphi提供了"靈巧調(diào)用(smart callback)",使得MakeProcInstance可以省去,而直接將鉤子過濾函數(shù)名用作入口地址。
這樣當(dāng)應(yīng)用程序?_c3GetMessage或PeekMessage函數(shù)從消息隊列中讀消息或有按鍵消息(WM_KEYDOWN或WM_KEYUP)要處理時,系統(tǒng)就要調(diào)用鉤子過濾函數(shù)KeyHookProc處理鍵盤消息。
4、 卸載鉤子過濾函數(shù)。
當(dāng)鉤子函數(shù)不再需要時,應(yīng)調(diào)用UnHookWindowsHookProc卸載安裝的鉤子以釋放系統(tǒng)資源。
完整的程序清單如下?_ba
Library KEYHOOK;
uses Windows;
const BUFFER_SIZE=16*1024;
const HOOK_MEM_FILENAME='SAMPLE KEY_HOOK_MEM_FILE';
const HOOK_MUTEX_NAME ='SAMPLE KEY_HOOK_MUTEX_NAME';
type
TShared=record
Keys : array[0..BUFFER_SIZE] of Char;
KeyCount : Integer;
end;
PShared=^TShared;
var
MemFile,HookMutex : THandle;
hOldKeyHook : HHook;
ProcSaveExit : Pointer;
Shared : PShared;

//鍵盤鉤子過濾函數(shù)
function KeyHookProc(iCode: Integer; wParam: WPARAM ; lParam: LPARAM):LRESULT
; stdcall; export;
const KeyPressMask = $80000000;
begin
if iCode < 0 then
Result := CallNextHookEx(hOldKeyHook, iCode, wParam, lParam)
else begin
if ((lParam and KeyPressMask)= 0) then // 鍵按下
begin
Shared^.Keys[Shared^.KeyCount]:=Char(wParam and $00ff);
Inc(Shared^.KeyCount);
if Shared^.KeyCount>=BUFFER_SIZE-1 then Shared^.KeyCount:=0;
end;
iCode:=-1;
Result := CallNextHookEx(hOldKeyHook, iCode, wParam, lParam);
end;
end;

// 設(shè)置鉤子過濾函數(shù)
function EnableKeyHook : BOOL ; export;
begin
Shared^.KeyCount:=0; //初始化鍵盤指針
if hOldKeyHook=0 then begin
hOldKeyHook := SetWindowsHookEx(WH_KEYBOARD,
KeyHookProc,
HInstance,
0);
end;
Result := (hOldKeyHook <> 0);
end;

//撤消鉤子過濾函數(shù)
function DisableKeyHook: BOOL ; export;
begin
if hOldKeyHook<> 0 then
begin
UnHookWindowsHookEx(hOldKeyHook); // 解除 Keyboard Hook
hOldKeyHook:= 0;
Shared^.KeyCount:=0;
end;
Result := (hOldKeyHook = 0);
end;

//取得鍵盤緩沖區(qū)中擊鍵的個數(shù)
function GetKeyCount :Integer ; export;
begin
Result:=Shared^.KeyCount;
end;

//取得鍵盤緩沖區(qū)的鍵
function GetKey(index:Integer) : Char ; export;
begin
Result:=Shared^.Keys[index];
end;

//清空鍵盤緩沖區(qū)
procedure ClearKeyString ; export;
begin
Shared^.KeyCount:=0;
end;

//DLL的退出處理過程
procedure KeyHookExit; far;
begin
if hOldKeyHook <> 0 then DisableKeyHook;
UnMapViewOfFile(Shared); // 釋放內(nèi)存映象文件
CloseHandle(MemFile); // 關(guān)閉映象文件
ExitProc := ProcSaveExit;
end;

exports // 定義輸出函數(shù)
EnableKeyHook,
DisableKeyHook,
GetKeyCount,
ClearKeyString,
GetKey;

begin
// DLL 初始化部分
HookMutex:=CreateMutex(nil,True,HOOK_MUTEX_NAME);
// 通過建立內(nèi)存映象文件以共享內(nèi)存
MemFile:=OpenFileMapping(FILE_MAP_WRITE,False,
HOOK_MEM_FILENAME);
if MemFile=0 then
MemFile:=CreateFileMapping($FFFFFFFF,nil,PAGE_READWRITE,0,
SizeOf(TShared) ,HOOK_MEM_FILENAME);
Shared:=MapViewOfFile(MemFile,File_MAP_WRITE,0,0,0);
ReleaseMutex(HookMutex);
CloseHandle(HookMutex);
ProcSaveExit := ExitProc; // 保存DLL的ExitProc
ExitProc := @KeyHookExit; // 設(shè)置DLL新的ExitProc
end.
      // 源代碼結(jié)束

二、 在自己的程序中使用編制好的鍵盤鉤子過濾函數(shù)。
鉤子函數(shù)編制好后,使用起來其實很簡單:首先調(diào)用SetWindowsHookEx安裝自己的鉤子過濾函數(shù),同時保存原先的鉤子過濾函數(shù)地址。這時鉤子函數(shù)就開始起作用了,它將按照你的要求處理鍵盤消息。程序運行完畢或不再需要監(jiān)視鍵盤消息時,調(diào)用UnHookWindowsHookProc函數(shù)卸載所安裝的鉤子函數(shù),同時恢復(fù)原來的鉤子過濾函數(shù)地址。
下面就是使用在以上編制的鉤子函數(shù)的例子:
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
StdCtrls, ExtCtrls;
type
TForm1 = class(TForm)
Memo1: TMemo;
Panel1: TPanel;
bSetHook: TButton;
bCancelHook: TButton;
bReadKeys: TButton;
bClearKeys: TButton;
Panel2: TPanel;
procedure bSetHookClick(Sender: TObject);
procedure bCancelHookClick(Sender: TObject);
procedure bReadKeysClick(Sender: TObject);
procedure bClearKeysClick(Sender: TObject);
end;
var Form1: TForm1;

implementation
{$R *.DFM}
function EnableKeyHook : BOOL ; external 'KEYHOOK.DLL';
function DisableKeyHook : BOOL ; external 'KEYHOOK.DLL';
function GetKeyCount : Integer ; external 'KEYHOOK.DLL';
function GetKey(idx:Integer) : Char ; external 'KEYHOOK.DLL';
procedure ClearKeyString ; external 'KEYHOOK.DLL';

procedure TForm1.bSetHookClick(Sender: TObject); // 設(shè)置鍵盤鉤 7ó
begin
EnableKeyHook;
bSetHook.Enabled :=False;
bCancelHook.Enabled:=True;
bReadKeys.Enabled :=True;
bClearKeys.Enabled :=True;
Panel2.Caption:=' 鍵盤鉤子已經(jīng)設(shè)置';
end;

procedure TForm1.bCancelHookClick(Sender: TObject); // 卸載鍵盤鉤子
begin
DisableKeyHook;
bSetHook.Enabled :=True;
bCancelHook.Enabled:=False;
bReadKeys.Enabled :=False;
bClearKeys.Enabled :=False;
Panel2.Caption:=' 鍵盤鉤子沒有設(shè)置';
end;

procedure TForm1.bReadKeysClick(Sender: TObject); // 取得擊鍵的歷史記錄
var i:Integer;
begin
Memo1.Lines.Clear; // 在Memo1中顯示擊鍵歷史記錄
for i:=0 to GetKeyCount-1 do
Memo1.Text:=Memo1.Text+GetKey(i);
end;

procedure TForm1.bClearKeysClick(Sender: TObject); // 清除擊鍵歷史記錄
begin
Memo1.Clear;
ClearKeyString;
end;

end.
// 源代碼結(jié)束

三、 Windows95下DLL中實現(xiàn)共享內(nèi)存
  在上面的鉤子函數(shù)所在的DLL文件中,需要使用共享內(nèi)存,即,所有擊鍵的記錄存儲在同一個數(shù)據(jù)段中。為什么要這樣做呢?這是因為Windows95的DLL調(diào)用方法與Windows3.X的方法不同。每個進(線)程在登錄某動態(tài)連接庫時都會為該動態(tài)連接庫傳入一個新的實例句柄(即DLL數(shù)據(jù)段的句柄)。這使得DLL各個實例之間互不干擾,但是這對那些所有DLL實例共享一組變量帶來一些困難。為了解決這個問題,我們在這兒通過建立內(nèi)存映射文件的方法來解決。即使用Windows的OpenFileMapping、CreateFileMapping和
MapViewOfFile三個函數(shù)來實現(xiàn)。使用方法如下:

MemFile是THandle類型,Shared是指針類型,HOOK_MEM_FILENAME是一常量串

MemFile:=OpenFileMapping(FILE_MAP_WRITE,False,
HOOK_MEM_FILENAME); //打開內(nèi)存映射文件
if MemFile=0 then //打開失敗則?_c2建內(nèi)存映射文件
MemFile:=CreateFileMapping($FFFFFFFF,nil,PAGE_READWRITE,0,
SizeOf(TShared) ,HOOK_MEM_FILENAME);
//映射文件到變量
Shared:=MapViewOfFile(MemFile,File_MAP_WRITE,0,0,0);

到此為止,你已經(jīng)知道用Delphi編制鉤子函數(shù)有多么容易。最后不得不提醒大家:鉤子函數(shù)雖然功能比較強,但如果使用不當(dāng)將會嚴(yán)重影響系統(tǒng)的效率,所以要盡量避免使用系統(tǒng)鉤子。非要使用不可時也應(yīng)該格外小心,應(yīng)使之盡可能小地影響系統(tǒng)的運行。
[全文完]

上一篇:在Delphi的DBGrid中插入其他可視組件

下一篇:將面向?qū)ο蟮乃枷胴灤┦冀K--談Delphi開發(fā)(第二篇)

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
學(xué)習(xí)交流
熱門圖片

新聞熱點

疑難解答

圖片精選

網(wǎng)友關(guān)注

主站蜘蛛池模板: 97久久超碰国产精品电影 | 亚洲 中文 欧美 日韩 在线观看 | 国产探花在线精品一区二区 | 成人国产精品一级毛片视频 | 日韩免费av一区二区 | 毛片在线免费 | 国产欧美精品一区二区 | 黄18网站| 精品国产乱码久久久久久蜜柚 | 久草精品在线观看 | 一本色道久久综合亚洲精品按摩 | 国产二区视频 | 亚洲黄色av网站 | 日韩免费在线观看视频 | 国产成人精品免高潮在线观看 | 欧美日韩免费一区 | 精品亚洲视频在线 | 久久亚洲国产精品日日av夜夜 | 成人免费视频网址 | 精品国产黄a∨片高清在线 毛片国产 | 精品国产一区二区三区性色av | 久久久精彩视频 | 亚洲一区二区三区在线播放 | 久久女同互慰一区二区三区 | 国产午夜精品久久久久久久 | 在线观看视频91 | 国产电影一区二区三区图片 | 欧美午夜精品理论片a级按摩 | 在线a级 | a级在线观看免费 | www.se天堂| 亚洲福利一区 | 国产精品一区久久久久 | 久久久久久久免费 | 草逼视频免费观看 | 青草视频在线观看免费 | 蜜桃久久久 | 青草精品| 精品国产不卡一区二区三区 | 女人口述交换啪啪高潮过程 | 日韩和的一区二区 |