是否想為你的Windows加上一雙眼睛,察看使用者在機器上所做的各種操作(例如建立、刪除檔案;改變檔案或目錄名字)呢?
這裡介紹一種利用Windows未公開函式實現這個功能的方法。
在Windows下有一個未公開函式SHChangeNotifyRegister可以把你的視窗新增到系統的系統訊息監視鏈中,該函式在Delphi中的定義如下:
Function SHChangeNotifyRegister(hWnd,uFlags,dwEventID,uMSG,cItems:LongWord;
lpps:PIDLSTRUCT):integer;stdcall;external 'Shell32.dll' index 2;
其中引數hWnd定義了監視系統操作的視窗得控制程式碼,引數uFlags dwEventID定義監視操作引數,引數uMsg定義操作訊息,引數cItems定義附加引數,引數lpps指定一個PIDLSTRUCT結構,該結構指定監視的目錄。
當函式呼叫成功之後,函式會返回一個監視操作控制程式碼,同時系統就會將hWnd指定的視窗加入到操作監視鏈中,當有檔案操作發生時,系統會向hWnd傳送uMsg指定的訊息,我們只要在程式中加入該訊息的處理函式就可以實現對系統操作的監視了。
如果要退出程式監視,就要呼叫另外一個未公開得函式SHChangeNotifyDeregister來取消程式監視。
下面是使用Delphi編寫的具體程式實現範例,首先建立一個新的工程檔案,然後在Form1中加入一個Button控制元件和一個Memo控制元件,
程式的程式碼如下:
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
StdCtrls,shlobj,Activex;
const
SHCNE_RENAMEITEM = $1;
SHCNE_CREATE = $2;
SHCNE_DELETE = $4;
SHCNE_MKDIR = $8;
SHCNE_RMDIR = $10;
SHCNE_MEDIAINSERTED = $20;
SHCNE_MEDIAREMOVED = $40;
SHCNE_DRIVEREMOVED = $80;
SHCNE_DRIVEADD = $100;
SHCNE_NETSHARE = $200;
SHCNE_NETUNSHARE = $400;
SHCNE_ATTRIBUTES = $800;
SHCNE_UPDATEDIR = $1000;
SHCNE_UPDATEITEM = $2000;
SHCNE_SERVERDISCONNECT = $4000;
SHCNE_UPDATEIMAGE = $8000;
SHCNE_DRIVEADDGUI = $10000;
SHCNE_RENAMEFOLDER = $20000;
SHCNE_FREESPACE = $40000;
SHCNE_ASSOCCHANGED = $8000000;
SHCNE_DISKEVENTS = $2381F;
SHCNE_GLOBALEVENTS = $C0581E0;
SHCNE_ALLEVENTS = $7FFFFFFF;
SHCNE_INTERRUPT = $80000000;
SHCNF_IDLIST = 0;
// LPITEMIDLIST
SHCNF_PATHA = $1;
// path name
SHCNF_PRINTERA = $2;
// printer friendly name
SHCNF_DWORD = $3;
// DWORD
SHCNF_PATHW = $5;
// path name
SHCNF_PRINTERW = $6;
// printer friendly name
SHCNF_TYPE = $FF;
SHCNF_FLUSH = $1000;
SHCNF_FLUSHNOWAIT = $2000;
SHCNF_PATH = SHCNF_PATHW;
SHCNF_PRINTER = SHCNF_PRINTERW;
WM_SHNOTIFY = $401;
NOERROR = 0;
type
TForm1 = class(TForm)
Button1: TButton;
Memo1: TMemo;
procedure FormClose(Sender: TObject; var Action: TCloseAction);
procedure Button1Click(Sender: TObject);
procedure FormCreate(Sender: TObject);
private
{ Private declarations }
procedure WMShellReg(var Message:TMessage);message WM_SHNOTIFY;
public
{ Public declarations }
end;
type PSHNOTIFYSTRUCT=^SHNOTIFYSTRUCT;
SHNOTIFYSTRUCT = record
dwItem1 : PItemIDList;
dwItem2 : PItemIDList;
end;
Type PSHFileInfoByte=^SHFileInfoByte;
_SHFileInfoByte = record
hIcon :Integer;
iIcon :Integer;
dwAttributes : Integer;
szDisplayName : array [0..259] of char;
szTypeName : array [0..79] of char;
end;
SHFileInfoByte=_SHFileInfoByte;
Type PIDLSTRUCT = ^IDLSTRUCT;
_IDLSTRUCT = record
pidl : PItemIDList;
bWatchSubFolders : Integer;
end;
IDLSTRUCT =_IDLSTRUCT;
function SHNotify_Register(hWnd : Integer) : Bool;
function SHNotify_UnRegister:Bool;
function SHEventName(strPath1,strPath2:string;lParam:Integer):string;
Function SHChangeNotifyDeregister(hNotify:integer):integer;stdcall;
external 'Shell32.dll' index 4;
Function SHChangeNotifyRegister(hWnd,uFlags,dwEventID,uMSG,cItems:LongWord;
lpps:PIDLSTRUCT):integer;stdcall;external 'Shell32.dll' index 2;
Function SHGetFileInfoPidl(pidl : PItemIDList;
dwFileAttributes : Integer;
psfib : PSHFILEINFOBYTE;
cbFileInfo : Integer;
uFlags : Integer):Integer;stdcall;
external 'Shell32.dll' name 'SHGetFileInfoA';
var
Form1: TForm1;
m_hSHNotify:Integer;
m_pidlDesktop : PItemIDList;
implementation
{$R *.DFM}
function SHEventName(strPath1,strPath2:string;lParam:Integer):string;
var
sEvent:String;
begin
case lParam of //根據引數設定提示訊息
SHCNE_RENAMEITEM: sEvent := '重新命名檔案'+strPath1+'為'+strpath2;
SHCNE_CREATE: sEvent := '建立檔案 檔名:'+strPath1;
SHCNE_DELETE: sEvent := '刪除檔案 檔名:'+strPath1;
SHCNE_MKDIR: sEvent := '新建目錄 目錄名:'+strPath1;
SHCNE_RMDIR: sEvent := '刪除目錄 目錄名:'+strPath1;
SHCNE_MEDIAINSERTED: sEvent := strPath1+'中插入可移動儲存介質';
SHCNE_MEDIAREMOVED: sEvent := strPath1+'中移去可移動儲存介質'+strPath1+' '+strpath2;
SHCNE_DRIVEREMOVED: sEvent := '移去驅動器'+strPath1;
SHCNE_DRIVEADD: sEvent := '新增驅動器'+strPath1;
SHCNE_NETSHARE: sEvent := '改變目錄'+strPath1+'的共享屬性';
SHCNE_ATTRIBUTES: sEvent := '改變檔案目錄屬性 檔名'+strPath1;
SHCNE_UPDATEDIR: sEvent := '更新目錄'+strPath1;
SHCNE_UPDATEITEM: sEvent := '更新檔案 檔名:'+strPath1;
SHCNE_SERVERDISCONNECT: sEvent := '斷開與伺服器的連線'+strPath1+' '+strpath2;
SHCNE_UPDATEIMAGE: sEvent := 'SHCNE_UPDATEIMAGE';
SHCNE_DRIVEADDGUI: sEvent := 'SHCNE_DRIVEADDGUI';
SHCNE_RENAMEFOLDER: sEvent := '重新命名資料夾'+strPath1+'為'+strpath2;
SHCNE_FREESPACE: sEvent := '磁碟空間大小改變';
SHCNE_ASSOCCHANGED: sEvent := '改變檔案關聯';
else
sEvent:='未知操作'+IntToStr(lParam);
end;
Result:=sEvent;
end;
function SHNotify_Register(hWnd : Integer) : Bool;
var
ps:PIDLSTRUCT;
begin
{$R-}
Result:=False;
If m_hSHNotify = 0 then begin
//獲取桌面資料夾的Pidl
if SHGetSpecialFolderLocation(0, CSIDL_DESKTOP,
m_pidlDesktop)<> NOERROR then
Form1.close;
if Boolean(m_pidlDesktop) then begin
ps.bWatchSubFolders := 1;
ps.pidl := m_pidlDesktop;
// 利用SHChangeNotifyRegister函式註冊系統訊息處理
m_hSHNotify := SHChangeNotifyRegister(hWnd, (SHCNF_TYPE Or SHCNF_IDLIST),
(SHCNE_ALLEVENTS Or SHCNE_INTERRUPT),
WM_SHNOTIFY, 1, ps);
Result := Boolean(m_hSHNotify);
end
Else
// 如果出現錯誤就使用 CoTaskMemFree函式來釋放控制程式碼
CoTaskMemFree(m_pidlDesktop);
End;
{$R+}
end;
function SHNotify_UnRegister:Bool;
begin
Result:=False;
If Boolean(m_hSHNotify) Then
//取消系統訊息監視,同時釋放桌面的Pidl
If Boolean(SHChangeNotifyDeregister(m_hSHNotify)) Then begin
{$R-}
m_hSHNotify := 0;
CoTaskMemFree(m_pidlDesktop);
Result := True;
{$R-}
End;
end;
procedure TForm1.WMShellReg(var Message:TMessage); //系統訊息處理函式
var
strPath1,strPath2:String;
charPath:array[0..259]of char;
pidlItem:PSHNOTIFYSTRUCT;
begin
pidlItem:=PSHNOTIFYSTRUCT(Message.wParam);
//獲得系統訊息相關得路徑
SHGetPathFromIDList(pidlItem.dwItem1,charPath);
strPath1:=charPath;
SHGetPathFromIDList(pidlItem.dwItem2,charPath);
strPath2:=charPath;
Memo1.Lines.Add(SHEvEntName(strPath1,strPath2,Message.lParam)+chr(13)+chr(10));
end;
procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
begin
//在程式退出的同時刪除監視
if Boolean(m_pidlDesktop) then
SHNotify_Unregister;
end;
procedure TForm1.Button1Click(Sender: TObject); //Button1的Click訊息
begin
m_hSHNotify:=0;
if SHNotify_Register(Form1.Handle) then begin //註冊Shell監視
ShowMessage('Shell監視程式成功註冊');
Button1.Enabled := False;
end
else
ShowMessage('Shell監視程式註冊失敗');
end;
procedure TForm1.FormCreate(Sender: TObject);
begin
Button1.Caption := '開啟監視';
end;
end.
執行程式,點選“開啟監視”按鈕,如果出現一個顯示“Shell監視程式成功註冊”的對話方塊,說明Form1已經加入到系統操作監視鏈中了,你可以試著在資源管理器中建立、刪除資料夾,移動檔案等操作,你可以發現這些操作都被記錄下來並顯示在文字框中。
在上面的程式中多次使用到了一個PItemIDList的結構,這個資料結構指定Windows下得一個“專案”,在Windows下資源實現統一管理一個“專案”可以是一個檔案或者一個資料夾,也可以是一個印表機等資源。另外一些API函式也涉及到了Shell(Windows外殼)操作,各位讀者可以參考相應的參考資料。
由於使用到了Windows的未公開函式,沒有相關得參考資料,所以有一些未知得操作(在Memo1中會顯示“未知操作”)。如果哪位讀者有興趣, http://member.netease.com/~blackcat 有實現該功能的VB程式下載。
以上程式在Windows98、Windows2000、Delphi5下執行通過。
看看這個合不合你用吧
是否想為你的Windows加上一雙眼睛,察看使用者在機器上所做的各種操作(例如建立、刪除檔案;改變檔案或目錄名字)呢?...
相關文章
- linux 刪除檔案或目錄——rmLinux
- Linux rm命令:刪除檔案或目錄Linux
- Linux rm(刪除檔案/目錄) 命令Linux
- 為什麼刪除記錄表檔案不會減小?(記錄的插入與刪除在磁碟上的變化)
- 電腦上怎麼徹底刪除一個檔案?兩種可以直接徹底刪除檔案的操作方法
- 刪除指定目錄下指定字尾的檔案
- linux 刪除問題 一次刪除多個目錄下的相同副檔名的檔案Linux
- #Java教程:File類:檔案、目錄的建立、刪除、判斷、getXXX、canXXX @FDDLCJava
- [20180707]刪除結尾帶.的檔案與目錄.txt
- linux伺服器目錄檔案的命令操作Linux伺服器
- php檔案操作之提取檔案/目錄的名稱PHP
- Python 檔案、目錄操作Python
- 在 Mac Catalina 的 root 目錄建立虛擬檔案Mac
- PHP 檔案操作的各種姿勢PHP
- c# 對檔案的各種操作C#
- 刪除當前目錄下的所有可執行檔案
- 用DeBug的方式,帶你掌握HBase檔案在Snapshot的各種變化
- 在 Linux 上壓縮檔案:zip 命令的各種變體及用法Linux
- 在Linux中,如何建立、刪除和修改檔案?Linux
- Linux基楚操作指引【檔案改名、檔案備份、檔案刪除】Linux
- Linux 使用 inotify 監控檔案或目錄變化Linux
- Linux如何在目錄下靈活建立、瀏覽、刪除百萬個檔案Linux
- Java SSH 客戶端 如何刪除 linux 機器上的 檔案Java客戶端Linux
- Linux學習初期,怎麼給檔案或目錄命名呢?Linux
- windows.old可以刪除嗎?windows.old檔案的刪除方法Windows
- 關於檔案系統在建立目錄檔案和普通檔案時的區別
- hadoop 資料夾檔案的建立與刪除Hadoop
- linux|批量建立檔案、資料夾或刪除——萬用字元Linux字元
- 刪除某一資料夾或檔案時,提示“操作無法完成,因為其中的資料夾或檔案已在另一個程式中開啟”
- ln 覆蓋普通檔案或目錄
- 恆訊科技教您如何用Linux系統用命令刪除一個檔案或者目錄呢?Linux
- LINUX 使用批量刪除檔案的幾種方法Linux
- 以前的windows安裝檔案可以刪除嗎 windowsold檔案刪除了有影響嗎Windows
- 【知識分享】linux伺服器目錄檔案的命令操作Linux伺服器
- Windows 10刪除備份檔案方法Windows
- 刪除vmvare的大檔案
- nodejs獲取檔案資訊,判斷是檔案或目錄NodeJS
- windows 把指定目錄的檔案複製到當前目錄Windows