京公网安备 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
主讲人简介 张冲,海归统计学硕士,CDA 认证数据分析师,前云南白药集团资深数据分析师,自媒体 Python 讲师,全网课程播放量破 ...
2026-04-10在数据可视化与业务分析中,同比分析是衡量业务发展趋势、识别周期波动的核心手段,其核心逻辑是将当前周期数据与上年同期数据进 ...
2026-04-10在机器学习模型的落地应用中,预测精度并非衡量模型可靠性的唯一标准,不确定性分析同样不可或缺。尤其是在医疗诊断、自动驾驶、 ...
2026-04-10数据本身是沉默的,唯有通过有效的呈现方式,才能让其背后的规律、趋势与价值被看见、被理解、被运用。统计制图(数据可视化)作 ...
2026-04-10在全球化深度发展的今天,跨文化传播已成为连接不同文明、促进多元共生的核心纽带,其研究核心围绕“信息传递、文化解读、意义建 ...
2026-04-09在数据可视化领域,折线图是展示时序数据、趋势变化的核心图表类型之一,其简洁的线条的能够清晰呈现数据的起伏规律。Python ECh ...
2026-04-09在数据驱动的时代,数据分析早已不是“凭经验、靠感觉”的零散操作,而是一套具备固定逻辑、标准化流程的系统方法——这就是数据 ...
2026-04-09长短期记忆网络(LSTM)作为循环神经网络(RNN)的重要改进模型,凭借其独特的门控机制(遗忘门、输入门、输出门),有效解决了 ...
2026-04-08在数据分析全流程中,数据质量是决定分析结论可靠性的核心前提,而异常值作为数据集中的“异类”,往往会干扰统计检验、模型训练 ...
2026-04-08在数字经济飞速发展的今天,数据已渗透到各行各业的核心场景,成为解读趋势、优化决策、创造价值的核心载体。而数据分析,作为挖 ...
2026-04-08在数据分析全流程中,数据处理是基础,图形可视化是核心呈现手段——前者负责将杂乱无章的原始数据转化为干净、规范、可分析的格 ...
2026-04-07在数据分析与统计推断中,p值是衡量假设检验结果显著性的核心指标,其本质是在原假设(通常为“无效应”“无差异”)成立的前提 ...
2026-04-07在数字经济深度渗透的今天,数据已成为企业生存发展的核心资产,企业的竞争本质已转变为数据利用能力的竞争。然而,大量来自生产 ...
2026-04-07Python凭借简洁的语法、丰富的生态库,成为算法开发、数据处理、机器学习等领域的首选语言。但受限于动态类型、解释性执行的特性 ...
2026-04-03在深度学习神经网络中,卷积操作是实现数据特征提取的核心引擎,更是让模型“看懂”数据、“解读”数据的关键所在。不同于传统机 ...
2026-04-03当数字化转型从企业的“战略口号”落地为“生存之战”,越来越多的企业意识到,转型的核心并非技术的堆砌,而是数据价值的深度挖 ...
2026-04-03在日常办公数据分析中,数据透视表凭借高效的汇总、分组功能,成为Excel、WPS等办公软件中最常用的数据分析工具之一。其中,“计 ...
2026-04-02在数字化交互的全场景中,用户的每一次操作都在生成动态的行为轨迹——电商用户的“浏览商品→点击详情→加入购物车”,内容APP ...
2026-04-02在数字化转型深度推进的今天,企业数据已成为驱动业务增长、构建核心竞争力的战略资产,而数据安全则是守护这份资产的“生命线” ...
2026-04-02在数据驱动决策的浪潮中,数据挖掘与数据分析是两个高频出现且极易被混淆的概念。有人将二者等同看待,认为“做数据分析就是做数 ...
2026-04-01