一种利用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 := '打开监视 ';

(0)

相关推荐

  • 利用Windows To Go实现随身定制化电脑

    很多人对电脑都有自己的使用习惯,但毕竟特殊情况或是其他一些问题,需要使用别人的电脑,这里小编就给大家一个方法,随便谁的电脑,只需要一个U盘或者移动硬盘,就能让它成为自己的定制化电脑.简单说,我们就是要 ...

  • 调用未公开的API函数的方法

    Windows SDK开发包中并未提供所有的API函数,在本文中我将讨论如何调用这种未公开的API函数。 事实上所有未公开的API函数都和其它的API函数一样包含在系统的动态链接库中,调用这些函数的方 ...

  • 利用Windows 7权限,间接提升系统安全性.

    在Windows XP系统下,用户在限制USB设备使用方面,首先考虑的就是禁止其自动播放,以避免木马病毒入侵。但很多时候,这样的方法并不是很有效。如果你是Windows 7用户,那就可以通过它独有的“ ...

  • Excel利用if和sum函数求单科成绩大于70分的人数

    小编就为大家介绍Excel利用if和sum函数求单科成绩大于70分的人数方法,利用if函数和sum函数,进行基本的函数运算,来看看吧! 步骤 数据如图所示 输入if函数,=if(K3>70,1, ...

  • 在Excel中利用IF函数对学生成绩进行评定的具体操作步骤

    我们都知道现在学校会保护学生的成绩隐私,直接根据分数把成绩分为"优"."良"."中"和"差"等几个级别,或者是" ...

  • 如何利用windows movie maker制作视频

    制作视频,其实很简单,今天我就来分享一下: 如何利用windows movie maker制作视频 操作方法 01 在主界面上点击"导入视频",导入我们所需要的视频. 02 把我们 ...

  • 如何利用windows自带工具清理系统垃圾

    电脑使用的过程中,会产生临时文件.缓存文件等垃圾信息,占用系统空间,拖慢系统运行速度.目前有很多软件可以帮助清理这些垃圾,在没有安装此类软件时,也可以方便的利用windows系统自带的小工具来清理垃圾 ...

  • 怎样在Excel中利用COUNTIF和COUNTA函数计算比例

    相信大家都对Excel有了了解,知道他能用来制作表格,用来统计数据,处理数据,等等一些其他应用,总之,他的应用时十分之广泛的,我们需要对他进行了解和掌握,才能利用Excel来为自己提供方便.下面介绍怎 ...

  • 如何利用excel中的函数求出中位数

    如何利用excel中的函数求出中位数呢?很多人都不会,下面小编来教大家. 操作方法 01 首先,我们打开我们电脑上面的excel,然后我们在表格中输入一些数字: 02 之后我们点击工具栏中的公式: 0 ...