在Visual Basic 中使用C++ 类

类别:VC语言 点击:0 评论:0 推荐:
在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 联机帮助文件。

本文地址:http://com.8s8s.com/it/it3484.htm