京公网安备 11010802034615号
经营许可证编号:京B2-20210330
实现进程间数据交换的两种方法和应用
Windows操作系统是一个多任务系统,每个任务都有相应的进程对应。熟悉windows系统的用户知道,每个进程都有自己独立的内存地址和内存空间。这对进程间之间的数据相互访问和相互交换带来一定的不便,但是在实际应用中有时要在进程间进行数据交换。windows系统提供了许多方法实现进程间的数据交换,比如我们可以通过磁盘文件来进行交换,这实际是最简单可行的办法,但运行速度大打折扣,特别是对于大批量数据交换时。另外我们可以通过Windows系统提供的CreateFileMaping、OpenFileMapping、MapViewofFile等API函数来实现进程间的数据交换,但实现方法相对复杂。 本文将介绍另外两种进程间数据交换的方法,一种是通过剪贴板的进程数据交换方法,另一种是直接内存读取的进程数据交换方法。最后将用Delphi语言举例加以实现应用。
一、通过剪贴板的进程数据交换方法 剪贴板是windows系统专门提供用来不同应用程序之间的数据交换,用户只要通过“复制”和“粘帖”两个动作就可以实现应用程序之间的数据交换。用户可以自定义剪贴板格式实现数据交换的目的。 其方法步骤为: 1、数据发送进程向剪贴板“复制”用户自定义的数据内容 2、数据发送进程向数据接受进程发送数据交换的消息。 3、数据接受进程收到消息以后向剪贴板“粘贴”用户自定义的数据内容。 这种方法的实现过程如图一所示:
该方法涉及到的重要API函数主要有: 1、RegisterClipboardFormat 这个函数是向系统剪贴板注册新的剪贴板数据格式其函数调用格式为:
UINT RegisterClipboardFormat(
LPCTSTR lpszFormat // 新剪贴板数据格式的名称
);
|
HGLOBAL GlobalAlloc(
UINT uFlags, //分配到的内存属性
DWORD dwBytes // 需要内存分配的空间大小
);
LPVOID GlobalLock(
HGLOBAL hMem // 需要加锁的内存“句柄”
);
|
该方法涉及的重要API函数主要有: 1、GetCurrentProcessId 这个函数主要获取当前运行进程的进程标示号,其函数格式为:
DWORD GetCurrentProcessId(VOID) |
HANDLE OpenProcess(
DWORD dwDesiredAccess, // 要打开的进程的访问模式
BOOL bInheritHandle, // 该进程句柄是否在子线程中继承
DWORD dwProcessId // 进程的标示号
);
|
BOOL ReadProcessMemory(
HANDLE hProcess, //进程句柄
LPCVOID lpBaseAddress, // 要读取内存的起始地址
LPVOID lpBuffer, //存放读取的数据的变量
DWORD nSize, //需要读取的字节数
LPDWORD lpNumberOfBytesRead // 实际读取到的字节数
);
|
TFriend=packed record
name :array[1..8] of char;
age :byte;
address :array[1..30] of char;
end;
|
1.WM_Datasend_byClpbrd=WM_user+1000; //用于通过剪贴板的进程数据交换的方法的消息; 2.WM_DataSend_byMemory=WM_user+1001;// 直接内存读取的进程数据交换方法的消息; 3.receivewindowcaption='DataReceiver'; //接收数据进程窗口的标题; 4.receivewindowclassname='Data Receive Window';//接受数据进程窗口的类别名称; 5.userDefineformat='process data exchange'; //自定义剪贴板数据格式名称; |
procedure TForm1.Button1Click(Sender: TObject);
var
wnd :Hwnd;
Friend :TFriend;
hmem :Thandle;
datapointer :pointer;
CF_User :Uint;
begin
//寻找数据接受进程的窗口
wnd:=findwindow(receivewindowclassname,receivewindowcaption);
//如果找到了向剪贴板复制数据并且向数据接受进程发送消息
if wnd<>0 then
begin
//申请内存
hmem:=globalAlloc(GMEM_MOVEABLE,sizeof(friend));
//给记录赋值
strpcopy(@friend.name,edit1.text);
friend.age:=byte(spinedit1.Value);
strpcopy(@friend.address,edit2.text);
//获取申请到的内存地址并且把记录的值导入内存
DataPointer := GlobalLock(hmem);
//把记录数据复制到Global内存中
move(friend,datapointer^,sizeof(friend));
//注册或获取新的剪贴板数据格式标号
CF_User:=registerclipboardformat(userdefineformat);
//向剪贴板复制数据
clipboard.Open;
clipboard.SetAsHandle(CF_USER,hmem);
clipboard.Close;
//向接收进程窗口发送消息
sendmessage(wnd,WM_Datasend_byClpbrd,0,0);
end;
end;
|
procedure TForm1.Button2Click(Sender: TObject);
var
wnd :Hwnd;
Friend :TFriend;
Begin
//寻找数据接受进程的窗口
wnd:=findwindow(receivewindowclassname,receivewindowcaption);
//如果找到了向数据接收进程窗口发送消息
if wnd<>0 then
begin
//对记录赋值
strpcopy(@friend.name,edit1.text);
friend.age:=byte(spinedit1.Value);
strpcopy(@friend.address,edit2.text);
//发送消息,把发送进程的进程标示号作为Wparam
//把记录地址作为Lparam进行传递
sendmessage(wnd,WM_DataSend_byMemory,
getcurrentprocessID,lparam(@Friend));
end;
end;
|
procedure TForm1.CreateParams(var Params: TCreateParams);
begin
inherited CreateParams(Params);
params.Caption:=receivewindowcaption;
Params.WinClassName :=receivewindowclassname;
end;
|
procedure Tform1.wndproc(var message:Tmessage);
var
processhnd :Thandle;
Friend :TFriend;
numread :dword;
hmem :Thandle;
clipdata :pointer;
CF_User :Uint;
Begin
Case message.msg of
//处理剪贴板方式发送过来的消息
WM_Datasend_byClpbrd:
Begin
//注册或获取新的剪贴板数据格式标号
CF_User:= registerclipboardformat(userdefineformat);
if clipboard.HasFormat(CF_User) then
begin
//获取剪贴板的内容,并且分别赋值给三个Edit
hmem:=clipboard.GetAsHandle(CF_user);
clipdata:=globallock(hmem);
if clipdata<>nil then
begin
move(clipdata^,Friend,sizeof(Friend));
edit1.text:=Friend.name;
edit2.text:=inttostr(Friend.age);
edit3.text:=Friend.address;
end;
end;
message.Result:=1;
End;
//处理直接内存读取发送过来的消息
WM_DataSend_byMemory:
Begin
//获取发送进程的“进程句柄”,设置访问模式为“读”
processhnd:=openprocess(PROCESS_VM_READ,
false,message.WParam);
//读取发送进程的内存数据
readprocessmemory(processhnd,
ptr(message.lparam),
@friend,sizeof(friend),numread);
edit1.text:=Friend.name;
edit2.text:=inttostr(Friend.age);
edit3.text:=Friend.address;
closehandle(processhnd);
message.result:=1;
End;
Else
//对于其它消息依然按照原来的方法进行消息处理
Inherited wndproc(message);
End;
|
图三以上程序仅仅是上面介绍的两种方法最简单应用,用户可以根据自己的实际需要加以拓展应用。上述程序在Windows 2000+Delphi 5.0的环境下运行通过
数据分析咨询请扫描二维码
若不方便扫码,搜微信号:CDAshujufenxi
在统计学分析、实验研究、业务数据复盘过程中,单因素方差分析是检验自变量对因变量是否存在显著影响的核心方法。其中,两个水平 ...
2026-05-26【核心关键词】算法、客户、大数据、互联网、调优、建模、模型优化、机器学习、评分卡模型、模型开发、智能风控、业务场景、数 ...
2026-05-26 很多数据分析师写过无数个 SELECT,但当被问到“新建一张表,该如何定义字段类型来保证数据质量”“创建视图和存储物理表有 ...
2026-05-26在数据清洗、统计分析与数据质量检测工作中,箱型图(又称箱线图、Box Plot)是最直观、最高效的可视化分析工具之一。相较于柱状 ...
2026-05-25在大数据分析、数据清洗、质量管控、风险监测等领域,异常数据识别是保障数据质量、确保分析结论精准、规避业务决策失误的核心基 ...
2026-05-25 很多数据分析师精通Excel函数和透视表,但当被问到“数据从哪里来”“表和视图有什么区别”“数据库管理系统和SQL是什么关系 ...
2026-05-25数字化经营时代,企业的市场竞争早已从经验决策转向数据决策。门店营收、用户转化、产品销量、成本损耗、存量资产等所有经营行为 ...
2026-05-22在MySQL数据库日常运维、业务数据校验、数据迁移与数据清洗场景中,自增主键ID的连续性校验是一项基础且关键的工作。MySQL的Auto ...
2026-05-22 很多企业团队并非缺乏指标,而是陷入“指标失控”:仪表盘上堆满实时跳动的数据,却无法回答“当前瓶颈在哪、下一步该做什么 ...
2026-05-22【核心关键词】大数据、可视化、存储、架构、客户、离线、产品、同步、实时、数据仓库、数据分析、数据可视化、存储数据、离线 ...
2026-05-21在电商流量红利消退、公域获客成本持续走高的当下,存量用户深度挖掘已成为店铺增收增效的核心抓手。相较于付费投放获取的陌生新 ...
2026-05-21 很多数据分析师每天盯着几十个指标,但当被问到“这套指标要支撑什么业务目标”“指标之间是什么逻辑关系”“业务变化时如何 ...
2026-05-21在数据驱动决策的时代,数据质量直接决定分析结果的可靠性与准确性,而异常值作为数据清洗中的核心痛点,往往会扭曲分析结论、误 ...
2026-05-20 很多数据分析师每天盯着GMV、DAU、转化率,但当被问到“哪些指标在所有行业都适用”“哪些指标只对电商有意义”“二者如何搭 ...
2026-05-20Agent的能力边界,很大程度上取决于其掌握的Skill质量和数量。传统做法是靠人工编写和维护Skill,但这条路很快会遇到瓶颈。业务 ...
2026-05-20在统计分析中,方差分析(ANOVA)是一种常用的假设检验方法,核心用于分析“一个或多个自变量对单个因变量的影响”,广泛应用于 ...
2026-05-19 很多数据分析师每天盯着GMV、DAU、转化率,但当被问到“什么是指标”“指标和维度有什么区别”“如何定义指标值的计算规则和 ...
2026-05-19想高效备考 CDA 一级,拒绝盲目刷题、冗余学习?《CDA 一级教材知识手册》重磅来袭!以官方教材为核心,浓缩 13 章 103 个核心考 ...
2026-05-19在数据统计分析中,卡方检验是一种常用的非参数检验方法,核心用于判断两个或多个分类变量之间是否存在显著关联,广泛应用于市场 ...
2026-05-18在企业数字化转型的浪潮中,很多企业陷入了“技术堆砌”的误区——上线了ERP、CRM、BI等各类系统,积累了海量数据,却依然面临“ ...
2026-05-18