| 我在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; Wr
 【责编:landss】 |