在Visual Basic 中使用C++ 类
用C++ Builder 创建可重用的OLE Automation
李国兴
我在Borland C++ For Windows 环境下编制工程辅肋设计程序时定义了一个生成AutoCAD DXF 图形交换文件的类,在后来用Visual Basic 编写另一辅肋设计程序时又需要生成DXF图形文件,为了利用已有的代码,我在Borland C++ Builder 3.0 中将普通的C++类转换成Automation Object 进程内服务器,供Visual basic 使用,成功地实现了不同语言源程序级的重用。现将转换方法简单地介绍给大家。供大家参考。
原C++ 类声明如下: (为节约篇幅,进行了大量简化只实现基本功能)
class Dxf{
private:
HFILE handle; //DXF 文件头
String SecStart; //节头
String SecEnd; //节尾
String Tables; //表节串
String Blocks; //块节串
String Entities; //实体节串
protected:
bool WriteTitleSec(); //写标题节
bool WriteBlockSec(); //写块节
bool WriteEntitiesSec(); //写实体节
public:
Dxf();
bool SaveToFile(char *filename); // 保存数据到文件
void Dxf_Line(float x1,float y1,float x2,float y2,int Layer=0); // 画直线
};
将C++类转换成进程内服务器的过程如下:
1. 打开BorLand C++ Builder 3.0 的集成开发环境,选择File菜单下的new 子菜单,C++ Builder 将打开项目选择窗口,选择ActiveX 页面,在Active 页面中选择Active Library。然后再选择File菜单下的new 子菜单,选择ActiveX 页面,在Active 页面中选择Automation object 打开Automation Object Wizard 对话框,在Class name 框中填入DxfCls 作为在VB中引用Dxf 类的类名,单点Ok 完成。C++ Builder 3.0 将建立类型库,完成工程的建立。并打开类型库编辑窗口。(一点说明,由于BorLand C++ Builder在建立Automation Object 时,缺省是建立进程外的自动化服务器,在此我仅需的是作为DLL的进程内的自动化服务器,因此要首先创造一外类型库,再把自动化服务器加到类型库中去。
2. 选择将工程以DxfCls 名存盘,并将unit1以Dxf存盘。
3. 在类型库编辑窗中的Attributes页中将Name 中的project1 改为Dxf,将Help组框中的Help string 改为dxf Library,此字符串将在VB 的引用选择窗中显示。其它不用修改。
4. 将原Dxf 类中的除构造和析构以外的函数暴露给VB,其方法是选择Idxfcls节点,单击工具栏上的Method,然后在Attributes 页上将name对话框中Method1修改为您所需要的函数名,再选择Parameters 页设置函数的返回数据类型和型参。在选择返回数据时,注意下拉列表框中不直接支持bool 作为返回类型,您可以选择short或者选择Varian_bool 作为返回类型,在这里我选择short来替代bool 作为返回类型。此外型参表中的前两项对应通常C函数中的数据名和数据类型,数据类型以下拉列表的型式选择。如果要设置函数的缺省形参可双击第三个参数,此时将弹出parameters Flags 编辑窗,在编辑窗中选中has default value ,在下面的编辑框中填入缺省数据。
5. 单击工具栏上的Refresh, 编辑器自动在您的Dxf.cpp和dxf.h中添加上您在类型库中设定的函数声明。
6. 打开dxf.cpp文件,您可在找到SaveToFile等函数的声明如下:
short STDMETHODCALLTYPE TDxfClsImpl::SaveToFile(LPSTR filename)
{
try
{
}
catch(Exception &e)
{
return Error(e.Message.c_str(), IID_IDxfCls);
}
return S_OK;
};
将原dxf.cpp中的已暴露的函数的实现部分粘贴到新的dxf.cpp 中的各函数的try 部分,并修改错误处理代码和返回数据。其它私有函数直接粘贴到新的文件中,仅修改函数名前的类名。
7. 打开原dxf.h文件,将类的声明部分中除已暴露的函数以外的其它部分粘贴到剪粘板,然后再打开当前工程的dxf.h窗口,将剪贴板的内容复制到TdxfClsImpl类中间。
8. 选择project菜单中的make dxfcls 将工程编译成DLL文件。至此完成了C++中的普通类到自动化服务器的迁移。
9. 选择Run 菜单下的Register ActiveX server, 对dxf 进行注册。以后就可以在VB中使用dxf 类了。
10. 在VB5.0中使用对象,在Visual Basic 项目中,选择工程|引用,打开引用窗口,选中dxf library 库,按确认后返回,然后打开视图|对象游览器,打开所有库下拉列表后选择dxf,再选择dxfcls就可以见到以上所暴露的SaveToFile和Dxf_line 函数。VB 工程中可以用Dim dxf as new dxfcls 创建引用了。新的dxf 实现类的声明和实现如下,头文件中的细体字是C++ Bulider 3.0 自动生成,粗体字是添加的。读者可与C++ Bulider 3.0 自动生成框架文件对照。 为了节约篇幅,自动生成的注释已去掉。
附一. Dxf.h 头文件
#ifndef dxfH
#define dxfH
#include <fcntl.h>
#include<io.h>
//---------------------------------------------------------------------------
#include "Dxf_TLB.h"
#define LIBID_DxfCls LIBID_Dxf
class ATL_NO_VTABLE TDxfClsImpl:
AUTOOBJECT_IMPL(TDxfClsImpl, DxfCls, IDxfCls)
{
public:
TDxfClsImpl();
BEGIN_COM_MAP(TDxfClsImpl)
AUTOOBJECT_COM_INTERFACE_ENTRIES(IDxfCls)
END_COM_MAP()
DECLARE_TYPED_COMSERVER_REGISTRY("Dxf.DxfCls")
private:
HFILE handle;
int Vports;
int LayerNum;
String SecStart;
String SecEnd;
String Tables;
String Blocks;
String Entities;
void WriteTitleSec();
void WriteBlockSec();
void WriteEntitiesSec();
protected:
short STDMETHODCALLTYPE SaveToFile(LPSTR filename);
STDMETHOD(Dxf_Line(float x1, float y1, float x2, float y2, short Layer));
};
#endif
附二. dxf.cpp文件
#include <vcl.h>
#pragma hdrstop
#include <stdio.h>
#include <atl\atlvcl.h>
#include "dxf.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
TDxfClsImpl::TDxfClsImpl()
{
SecStart=" 0\nSECTION\n";
SecEnd=" 0\nENDSEC\n";
Vports=2;
LayerNum=1;
Tables=" 2\n\TABLES\n 0\n\TABLE\n 2\n\VPORT\n\ 70\n";
Blocks=" 2\nBLOCKS\n";
Entities=" 2\n\ENTITIES\n";
}
void TDxfClsImpl::WriteTitleSec()
{
write(handle, SecStart.c_str(), SecStart.Length()); // 写节头
write(handle," 2\nHEADER\n",strlen(" 2\nHEADER\n"));
write(handle, SecEnd.c_str(),SecEnd.Length()); // 写节尾
}
void TDxfClsImpl::WriteBlockSec()
{
write(handle,SecStart.c_str(), SecStart.Length()); // 写节头
write(handle,Blocks.c_str(),Blocks.Length());
write(handle,SecEnd.c_str(),SecEnd.Length()); // 写节尾
}
void TDxfClsImpl::WriteEntitiesSec() // 写实体节
{
write(handle, SecStart.c_str(), SecStart.Length());
write(handle,Entities.c_str(),Entities.Length());
write(handle, SecEnd.c_str(),SecEnd.Length());
}
short STDMETHODCALLTYPE TDxfClsImpl::SaveToFile(LPSTR filename)
{
int i;
try {
if (FileExists(filename)){
i=FileGetAttr(filename);
if (i& faReadOnly)
FileSetAttr(filename,i&0xFE);
DeleteFile(filename); }
handle=open(filename,O_CREAT|O_TEXT|O_RDWR);
if (handle==-1) return false;
WriteTitleSec(); //写标题节
WriteBlockSec();
WriteEntitiesSec();
write(handle," 0\nEOF\n",strlen(" 0\nEOF\n"));
close(handle);
}
catch(Exception &e)
{
return false;
}
return true;
};
//---------------------------------------------------------------------------
STDMETHODIMP TDxfClsImpl::Dxf_Line(float x1, float y1, float x2, float y2,
short Layer)
{
char buffer[16];
try {
Entities+=" 0\nLINE\n 8\n";
Entities+=IntToStr(Layer);
Entities+="\n 10\n";
sprintf(buffer,"%6.1f",x1);
Entities+=buffer;
Entities+="\n 20\n";
sprintf(buffer,"%6.1f",y1);
Entities+=buffer;
Entities+="\n 30\n 0.0\n 11\n";
sprintf(buffer,"%6.1f",x2);
Entities+=buffer;
Entities+="\n 21\n";
sprintf(buffer,"%6.1f",y2);
Entities+=buffer;
Entities+="\n 31\n 0.0\n"; }
catch(Exception &e) {
return Error(e.Message.c_str(), IID_IDxfCls);
}
return S_OK;
};
附三 示例:
在VB 5 中建一新窗口,在窗口上放一CommandButton,命名为cmdCreate,然后输入以下代码,运行后,单击cmdCreate,就可生成基本的AutoCAD下使用的.DXF文件。
Private Sub cmdCreate_Click()
Dim Dxf As New DxfCls
Dxf.Dxf_Line 100, 100, 500, 100
Dxf.SaveToFile App.Path & "\dxftest.dxf"
End Sub
以上程序分别在Borland C++ Builder 3 Server/Client suit 和VB 5.0 专业版下运行通过。生成的dxftest.dxf在AutoCAD for windows 12 下显示图形正常。
一点体会:现在各种语言都提供了创建OLE Automation 服务器的功能,为我们重用以前的代码和利用各种语言进行混合编程提供了方便。我们可利用各种语言的特点,快速地开发出高效的应用软件。
参考:Borland C++ Builder 3 Server/Client suit 联机帮助文件。