DDE与COM技术在组态软件开发中的应用
2008-07-30
作者:常 青 张 卡 张志杰
摘 要: 介绍了VC和Matlab通过COM与DDE技术在组态软件" title="组态软件">组态软件开发中的应用方法,发挥了Matlab在数值计算中的强大功能和InTouch在人机对话" title="人机对话">人机对话界面开发中的独特效果。
关键词: Matlab InTouch COM DDE 组态软件
随着各类嵌入式系统和现场总线的蓬勃发展,组态软件越来越成为工业自动化系统中的灵魂。它在实时数据存储、检索和图形显示及人机对话等方面都具有独特的效果。但是组态软件的脚本语言非常简单,在数据处理、算法实现等方面相对薄弱。这从一定程度上限制了组态软件在工业自动化中的应用。
Matlab作为一款优秀的数值计算软件,提供了应用于信号处理、工业控制、应用数学等各个领域的工具箱,但是它对运行环境的要求非常高,而且占用了庞大的系统资源,生成实用的人机对话界面的能力不强。
为了开发出具有友好人机对话界面、实现多种控制算法并完成实时数字信号处理的组态软件,可以使用Matlab进行数值计算,将处理后的数据传输给组态软件进行人机对话界面开发,以此发挥它们各自的优点。这样,开发软件之间的数据交换和处理就成为问题的关键。
本文根据Matlab和应用非常广泛的组态软件InTouch提供的编程接口和数据通信协议,提出了一种使用DDE和COM技术进行组态软件开发" title="软件开发">软件开发的方法。
1 基本思路与系统构架
动态数据交换(DDE)是一个由Microsoft开发的通信协议。该协议允许在Windows环境中的应用程序" title="应用程序">应用程序之间彼此发送/接收数据和指令。它在两个同时运行的应用程序之间实现客户服务器关系。服务器应用程序提供数据并接收对这些数据感兴趣的其它应用程序的请求。发请求的应用程序叫做客户。InTouch可以利用Microsoft的DDE与其他Windows程序通信,并可同时作为客户或服务器程序。
Matlab提供了多种混合编程的方法,但是这些方法大都不能脱离Matlab的运行环境,也不方便其它应用程序调用。为了摆脱Matlab运行环境,并在其基础上进行功能模块设计,方便其它应用程序调用,MathWorks公司推荐使用COM builder在Matlab环境下开发COM。COM是Component Object Module的简称,它是一种通用的对象接口,任何语言只要按照这种接口标准就可以调用它。
这样,可以使用Matlab开发COM组件,在VC下开发DDE服务程序,使其与InTouch进行数据通信,而这个DDE服务程序调用Matlab开发的COM进行数值处理和算法实现。在实际工业自动化的组态软件开发中,可以使用VC进行数据的采集、命令的发送和复杂的流程控制;使用Matlab下开发的COM完成复杂的算法实现和数字信号处理编程;在InTouch下实现人机对话界面并接收使用者的命令。具体的系统构架如图1所示。
2 应用实现
Windows DDE功能应用的核心是DDE事务管理库(DDEML) ,它负责管理Windows操作系统下应用程序间的DDE会话和通信,还向用户提供了一系列的应用程序接口API函数。
DDE实现程序间的通信是通过三个标识约定的:
应用程序名(Application):进行DDE对话双方的名称;
主题(Topic):被讨论的数据域;
项目(Item):被讨论的特定数据对象。
在DDE服务程序中首先使用DdeInitialize进行初始化,然后调用DdeCreateStringHandle建立应用程序名、主题和项目等标识的句柄" title="句柄">句柄,再通过DdeNameService在操作系统中注册DDE服务程序的名字。根据这些句柄,客户程序就可以使用它提供的DDE服务了。在VC++中的程序实现如下:
#include
DWORD idInst = 0, iData; //iDate是项目对应的数据
HSZ hszSvr, hszTopic, hszItem;
DdeInitialize(&idInst,(PFNCALLBACK)DdeCallback, CBF_FAIL_EXECUTES | CBF_SKIP_ALLNOTIFICATIONS, 0L);
//建立应用程序名、主题和项目等标识的句柄
hszSvr=DdeCreateStringHandle(idInst,“DDEApp”, 0);
hszTopic=DdeCreateStringHandle(idInst,“DDEAppTopic”, 0);
hszItem=DdeCreateStringHandle(idInst,“DDEAppItem”, 0);
//在操作系统中注册该DDE服务
DdeNameService(idInst, hszSvr, 0L, DNS_REGISTER);
… …
DDE服务程序的核心部分是一个回调函数,它处理所有DDE消息及相应数据请求。DDE服务程序回调函数的代码如下:
HDDEDATA CALLBACK DdeCallback(WORD usType, WORD usFmt, HCONV hConv, HSZ hsz1, HSZ hsz2, HDDEDATA hData, DWORD lData1, DWORD lData2) {
CHAR sz[5];
if (usType == XTYP_CONNECT) { //得到连接请求
return((HDDEDATA)TRUE);
}
//校验主题、项目的句柄及数据格式
if ( hsz1==hszTopic && hsz2==hszItem && usFmt==
CF_TEXT){
if ( usType == XTYP_REQUEST || usType ==XTYP_
ADVREQ ) { //得到数据请求
_itoa( iData, sz, 10 ); //将数据转换为文本格式
return DdeCreateDataHandle(idInst, (LPBYTE)sz,
strlen(sz) + 1, 0L, hszItem, CF_TEXT, 0);
}
if (usType == XTYP_POKE) { //得到客户端发送来的数据
DdeGetData(hData, (LPBYTE)sz, strlen(sz) + 1, 0L);
iData = atoi( sz ); //保存数据
DdePostAdvise(idInst, hszTopic, hszItem);
return((HDDEDATA)DDE_FACK);
}
}
return 0;
}
在任何需要向DDE客户端发送数据时使用DdePostAdvise触发XTYP_ADVREQ,从而达到向客户程序发送数据的目的。
在InTouch的标记名字典中定义I/O类型变量,以此调用DDE服务程序发送过来的数据。在声明I/O类型的访问名时只要确定DDE服务程序的应用程序名、主题名和项目名即可。
组态软件中的数值计算和实时数字信号处理部分在Matlab中实现并以COM组件的方式提供。这个部分的关键是Matlab下M文件的编写、COM组件的形成和DDE服务程序中对COM的调用。
COM组件实际是一个C++类,但接口都是纯虚类,组件从接口派生而来。在Matlab下通过键入comtool启动combuilder,根据提示设置类的名字和一些其它选项。为这个类添加方法(methods)通过向工程中添加M文件实现。这个M文件不是脚本文件而是函数文件。添加属性(Properties)则是在M文件中通过Global定义。至于事件(events)则需要用到语法%#event。举例说明如下:
%mymethod.m文件源代码 %myevent.m文件源代码
function mymethod function myevent
global mValue; %#event
…… ……
将以上两个文件添加到这个工程中则添加了一个方法mymethod、属性mValue和事件myevent。通过编译生成一个dll文件。这个dll就是COM的发布形式。
在VC中调用此COM与调用其它COM是一样的,所不同的是需要在VC的工程中包含Matlab提供的文件,具体操作是在Include files中添加
Combuilder也提供了COM组件的打包工具,它生成一个自解压文件,通过它实现必要的DLL安装和COM注册。
在DDE服务程序中调用COM组件进行数值计算和信号处理并向InTouch提供DDE服务,在InTouch中通过调用DDE传来的数据充实人机对话界面并接收操作者的命令以完成系统的功能。
3 实际开发中细节问题的考虑
Matlab下的COM组件开发是Matlab6.5才有的功能,早期的版本并没有这个模块。另外,并不是所有的Matlab工具箱都支持COM编译,这时可以考虑使用其它的函数代替,也可以使用DDE调用Matlab函数,但是这样不能脱离Matlab的运行环境。
在组态软件中使用DDE和COM技术可以充分发挥Matlab在数值计算中的功能和InTouch在人机对话界面开发中的独特效果,适用于功能模块设计和大规模组态软件的开发。
参考文献
1 马国华. 监控组态软件及其应用.北京:清华大学出版社,2001
2 InTouch用户指南. Wonderware Corporation,1998
3 MATLAB COM Builder User's Guide. MathWorks,2002