QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 1847|回复: 0
打印 上一主题 下一主题

COM应用软件开发技术

[复制链接]
字体大小: 正常 放大
韩冰        

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

跳转到指定楼层
1#
发表于 2005-1-26 01:07 |只看该作者 |倒序浏览
|招呼Ta 关注Ta

蔡倩

7 y( l: z. y$ C$ {- f* Y

主题词: COM ActiveX C++ Builder

" j8 J' r. r7 m* r/ i. m* q4 y! @

1.COM技术概述

- R! k4 f% Y' H" r8 X/ [

COM表示Component Object Model(组件对象模型),它是Microsoft大力推广的软件开发技术。采用COM规范开发的应用软件具有强大的功能,主要有如下几点:

; ?& V$ J& S5 ?

◆COM是二进制编程规范,可以编写被多种语言使用的代码。

: ^, @7 x5 Q5 E& O

◆用于创建ActiveX控件。

+ D' z7 ]4 w1 T1 x* A; S( V

◆通过OLE Automation 控制其它的程序。

: ]8 r u2 o8 F$ o3 J

◆与其它机器上的对象或程序进行对话,构成分布式应用程序。

- z n$ H+ `% a. E4 _4 f

Microsoft推出Windows 98和Windows NT 5.0后,整个操作系统的核心都围绕着COM来建立。我们可以把Windows系统看作是一系列的COM接口,在需要是可以调用这些接口。如DirectX就是一系列的COM接口服务程序,通过它可以进行高性能的Windows图形程序设计。

6 J/ s+ z' o T+ [

用COM技术开发的应用程序从理论上说是客户/服务器模式的程序。程序员可以使用一系列的COM服务程序来构造他们自己的应用程序,这些服务程序可以根据需要随时嵌入到主程序中。在分布式系统中,可以通过网络来访问这些服务程序。将来,操作系统和整个网络可能会被看作是一套以COM对象形式提供的服务集。一部分程序员负责建立这些服务,而另一部分程序员只负责如何调用它们。其目的是实现软件的即插即用。

y F# ~9 F6 r* O3 W. T6 H1 {& `2 Y

开发COM应用程序是比较复杂的,通常需采用ActiveX模板库(ATL)来编程。在这里我们推荐采用C++ Builder来开发COM程序,Inprise(Borland)公司的面向对象技术一直处于世界领先水平,C++ Builder采用可视化方法,隐藏了ATL的实现细节,自动生成COM接口所需的代码。

, {$ M2 u* m3 ?( v+ B% w( w) A

以下的程序举例采用C++ Builder 4.0 编制,在中文Windows98环境下运行。

2 e Z& u) P% Q/ T: ^ O

2.建立COM服务程序

9 |7 [( N3 Z, r: u

COM服务程序有三种形式,第一种是驻留在本地机器上以DLL形式提供,该服务程序被调用时,嵌入到调用程序的线程中运行;第二种是驻留在本地机器上以EXE形式提供,该服务程序被调用时将占用独立的线程运行;第三种驻留在远端机器上以EXE形式提供,服务程序通过网络被调用,它在远端机器上运行,结果通过网络返回调用者。

{% p5 f! ]# L! B/ L5 _

在此采用第一种形式建立COM服务程序,这也是最常用的形式,DirectX就是采用这种形式提供的。

$ i2 L6 l8 K" Q+ t. C0 n9 n& a4 `% f

C++ Builder建立COM服务程序的方法如下:

- O; h; o3 O. Y1 w: C& l1 u

2.1创建支持COM接口对象的动态连接库文件:

' ~+ t# y# [. O

◆打开File/New/ActiveX项目页,选择ActiveX Library;

* D1 A. e6 R* \/ Y6 f, P: g" Y D

◆选择Save All 将项目以PCOMServer文件名保存;此时C++ Builder 自动生成如下的文件:

% v" q$ \. T1 Q% n

PCOMServer.bpr:工程的项目文件;

+ }2 Z3 Z3 n( H, J& D

PCOMServer.h,PCOMServer.cpp:支持COM对象的动态连接库源文件,其中有许多函数用于COM接口对象的自动装配,大家不用去编辑它们;

4 v }3 i4 @# K* K8 |. R

PCOMServer_ATL.h,PCOMServer_ATL.cpp:ATL形式的文件供C++ Builder编译器调用,大家也不要去编辑它们。

3 W2 ?3 w& w/ Y$ Z8 _4 e

◆打开Project/Options/Linker 属性页不选中Use dynamic RTL选项,打开Project/Options/Packages属性页不选中Builder with runtime packages选项,这两步操作可以使开发的COM动态连接库不依赖C++ Builder的VCL动态连接库,有利于独立发行,但在一般情况下还是建议选中这两项。

9 }, v" ^7 [, s' p

2.2建立COM接口对象

& h- e$ d7 ]6 ^9 { {% V- Q$ X: w- f

2 Z1 i& f x; g% x( \5 W

) t, x5 L b; u* X6 w$ c4 o

打开File/New/ActiveX属性页,选择Automation Object表示向服务程序中插入一个自动类型的COM对象,我们选择这种类型的COM对象是为了可以自动注册,并且自动支持可以被其他语言调用。此时出现如下的对话框,输入COM类的名字MyCOM即可,对话框中的其它选项用于规定COM对象的性质,可查看帮助信息。

, B# y* b% ]) {9 O/ F

2.3通过类型库编辑器编辑COM对象中相应接口对象的属性和方法

/ L) b0 r7 V' E

5 [. K+ t/ `8 t

6 _6 g; v1 F. e" r

此时自动进入类型库编辑器,类型库用于存储COM对象的说明,是一个可以被多种语言调用的头文件包。在类型库中,可以定义COM对象的接口,定义接口对象的属性和方法等。类型库编辑器如下所示:

! p( z. x- B: `; [+ G5 M

可以看出此时自动产生了MyCOM类的一个接口类IMyCOM,在COM应用软件中我们实际上是与接口对象打交道,下面通过类型库编辑器为IMyCOM接口定义方法和属性。

# Q1 p! h0 E9 ?8 W- e0 p! s

◆单击编辑器顶部的Method按钮;

7 @& g# w F: U

◆在Arributes页面的Name字段中输入方法的名称,本例中是AddInt用于整数加法;

6 J3 a# s H% j# v3 X) b, W* E

◆在Parameters页面中,单击Add按钮编辑方法中的参数;

1 a+ c$ y+ G. S, `$ s6 ]

x和y是输入的两个整数,ret用于返回运算的结果,必须定义为指针型

: l3 m5 C T' T; ], J

◆切换到Flags页面,可以对接口的属性作调整;

# q' z0 ^4 Y. U/ M1 H5 X2 z) p

◆在Text页面中可以检查生成的IDL代码:

2 n7 r" m2 y+ X2 Y {! b

[id(0x00000001)]

( S1 A* S9 @7 W- N% a9 \( y

# Y' ?- j/ ?- M' g& r

8 J9 p5 n& P: z: R* ~# U3 x

HRESULT _stdcall AddInt([in] int x, [in] int y, [out, retval] int * ret );

" _9 z) {: M; H" X

◆单击Refresh按钮,此时可以关闭类型库编辑器。当需要为接口添加新的属性和方法时,可以通过View/Type Library重新打开编辑器。选择Save All用C++ Builder提供的缺省文件名保存类型库的相关文件如下:

6 o3 H( p5 d, @& G" u- z

PCOMServer.TLB: 类型库文件;

1 ?+ F% n! j; {6 E4 i& e

PCOMServer_TLB.cpp:包含COM接口和对象的说明,其主要目的是方便访问,在客户程序中需将本文件包含到客户程序的工程中;

5 N. _+ ~: e7 ?$ ]7 W% @

PCOMServer_TLB.h: PCOMServer_TLB.cpp的头文件,通过#include引入到客户程序中。

. ~% e. G% Y! Z6 N$ A

MyCOMImpl.cpp: 该文件是我们需要编写程序代码的地方,实现类型库定义的接口对象的方法和属性;

+ h* X& J; ?8 r8 L* @, [5 \; I

MyCOMImpl.h: MyCOMImpl.cpp的头文件。

( P2 |$ D4 O5 A) k$ Q2 q3 `& B0 z

2.4 实现COM接口中的方法

' I8 D+ p9 }- A2 A

打开MyCOMImpl.cpp文件会发现我们在类型库编辑器中定义的方法,为该方法编写代码如下:

, V) V; X* y* {* w0 K( Q

STDMETHODIMP TMyCOMImpl::AddInt(int x, int y, int* ret)

! t% B8 Q4 q# A- u# y# R0 A

{

" Y J3 N8 q) a( o5 Q7 _% u" r

*ret=x+y;

! A3 m+ |! J* S: n% a5 n

return S_OK;

1 ^( s; D- z1 i1 o( s

}

- S% H$ r1 r% S1 N; |

2.5 生成DLL文件并注册COM对象

) n8 p/ g C% Y. ^, u3 W

◆选择Project/Builder PCOMServer 生成PCOMServer.DLL文件。

8 a; J; @( o- [

◆打开类型库编辑器,单击Register按钮完成对COM对象的注册。

3 U, D3 P( A7 _* S5 s

通过Windows任务栏中的Run菜单运行REGEDIT程序,在Windows注册表的HKEY_CLASSES_ROOT键下查找到PCOMServer.MyCOM子键,PCOMServer为DLL文件的名字,MyCOM为COM对象的名字,在下面可以看到该COM对象的全局唯一描述符CLSID如下:

1 f/ P/ ]0 j: l7 _7 s

{59834F03-49F1-11D3-B85B-00E09804A418}

% ?2 p& c h7 L" t& h0 S' x, \) r

注意:不同的机器生成的描述符不同.

6 X7 G. R9 b8 e! F3 d+ r

在HKEY_CLASSES_ROOT键下查找到CLSID子键,在它下面找{59834F03-49F1-11D3-B85B-00E09804A418}子键,下面有如下的条目:

, t9 x: ]$ G3 p6 ]+ S

InprocServer32:存储PCOMServer.DLL的路径目录;

% J; u5 c( T1 C6 ]1 F6 N) q

ProgID:存储COM对象的注册名:PCOMServer.MyCOM;

) I0 f, ^/ S6 F0 W; i- _

Typelib:存储COM对象的CLSID值{59834F03-49F1-11D3-B85B-00E09804A418}。

9 g! H# F. M5 T9 O

COM对象就是通过在注册表中的纪录实现DLL与客户程序的自动连接。

8 _% C) |/ r& w. {6 n+ ]

3.建立COM客户程序

& R& I5 S! k# k9 [$ \) \

客户程序将访问PCOMServer.DLL服务程序中的MyCOM对象,这些对象的说明保存在前面所述的TLB文件中。我们可以直接将PCOMServer_TLB.cpp加入到客户程序的项目文件中,并在客户程序中引用PCOMServer_TLB.h文件;也可以通过Project/Import Type Library引用PCOMServer_TLB.TLB文件,重新生成.cpp和.h文件,自动完成上述过程。

6 k/ c- Q3 A; n: B7 X9 d

客户程序的编程重点是实现对服务程序中COM对象的方法的调用,调用的方法有多种,都是通过所谓的代理接口来完成的,这些代理接口在PCOMServer_TLB.h中有详细的定义,从这些定义中可以看出这些代理接口调用对象方法的过程。

0 O9 a6 [/ u8 M

PCOMServer_TLB.h文件很重要,包含了调用MyCOM对象的各种接口信息,该文件主要内容如下:

! z- ?. s$ \8 B6 Q& S

// Type Lib: D:\CAI\com\PCOMServer.tlb

1 r( W& i' R9 g( B+ u

// IID\LCID: {5BD378E5-4B57-11D3-B85B-00E09804A418}\0

: }3 w$ U; K! m( Z4 x

// Helpfile:

' y# F5 N; r/ c1 d2 ]6 K7 U3 c

// DepndLst:

- I* R9 D. G2 e3 N1 j0 K

// (1) v2.0 stdole, (C:\WINDOWS\SYSTEM\STDOLE2.TLB)

0 l6 N" _& h: b% P4 h; q& s

// (2) v4.0 StdVCL, (C:\WINDOWS\SYSTEM\STDVCL40.DLL)

# D5 l% ^0 Y8 Y# u) W

// ************************************************************************

' ^* W5 {* i. I- R1 u g- B# N" Q

#ifndef __PCOMServer_TLB_h__

5 m( E: D* A ?+ ]) ~

#define __PCOMServer_TLB_h__

. C6 p- ^9 y& L/ Z- Q3 l+ H7 |

#pragma option push -b -w-inl

q) X8 k: Q8 L4 [3 `% t$ q: N

#include <vcl/utilcls.h>

( I( J4 u2 \2 \" |. q

#if !defined(__UTILCLS_H_VERSION) || (__UTILCLS_H_VERSION < 0x0101)

; A: r5 j0 Z/ K+ Z# g0 ]

#error "This file requires an newer version of the header file UTILCLS.H"

* f8 B3 M3 ~# F& o% l5 J

#endif

4 W0 j" Q0 B% {1 l

% t$ S/ l" H" X% l( p& ]% C

#include <olectl.h>

% L% B) _5 j$ }) `$ r2 Z2 M

#include <ocidl.h>

' ~1 {% q1 _. k- d. q) y6 X

#if defined(USING_ATLVCL) || defined(USING_ATL)

% G8 `* Q$ W2 e+ }! F

#if !defined(__TLB_NO_EVENT_WRAPPERS)

$ a; ~: F) @+ z! ], ?! ]

#include <atl/atlmod.h>

: L. s; g0 c7 `! c1 j( R

#endif

- C& Z4 H) O& ~' f

#endif

* E8 k! ~- v! L7 ?6 ], P4 l

% f. D, y1 t: K' t$ v3 s8 @% \

namespace Stdvcl {class IStrings; class IStringsDisp;}

- X3 Z8 l) t: c7 W: G D/ D

using namespace Stdvcl;

0 K ]1 Y' v& N- I' g

/ h- V$ f, b# J. G

namespace Pcomserver_tlb

6 B! K/ Z/ |. ]* g3 l+ C4 A

{

) ~4 k: M+ y+ I/ M- P0 J

DEFINE_GUID(LIBID_PCOMServer, 0x5BD378E5, 0x4B57, 0x11D3, 0xB8, 0x5B, 0x00, 0xE0, 0x98, 0x04, 0xA4, 0x18);

# w6 Q7 Y+ U! [4 e. ?

DEFINE_GUID(IID_IMyCOM, 0x5BD378E6, 0x4B57, 0x11D3, 0xB8, 0x5B, 0x00, 0xE0, 0x98, 0x04, 0xA4, 0x18);

& r( l) k5 m6 q( y

DEFINE_GUID(CLSID_MyCOM, 0x5BD378E8, 0x4B57, 0x11D3, 0xB8, 0x5B, 0x00, 0xE0, 0x98, 0x04, 0xA4, 0x18);

" f" u7 S/ x3 J+ R) C

interface DECLSPEC_UUID("{5BD378E6-4B57-11D3-B85B-00E09804A418}") IMyCOM;

4 Z5 k; H& ^/ n+ k7 ]3 `( m

typedef IMyCOM MyCOM;

4 Q' M$ N, E$ y: t% I. Y( v

1 X/ r4 N) o4 m" d( J

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

4 ^/ u0 D! j# i) b

interface IMyCOM : public IDispatch

8 |5 b3 w( `0 j' J) O4 |0 g5 Z

{

' z4 w3 O! b' }

public:

6 Y; X' K2 [$ l7 L

virtual HRESULT STDMETHODCALLTYPE AddInt(int x/*[in]*/, int y/*[in]*/, int* ret/*[out,retval]*/) = 0; // [1]

( n/ t3 Q" G- c

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

* X8 O9 K+ h2 w8 |0 m

int __fastcall AddInt(int x/*[in]*/, int y/*[in]*/)

1 G" ^, C7 z1 V

{

3 t) Y! \% Q3 k+ Z# q8 P

int ret;

1 s/ L9 R! }2 m, e, i0 s8 i& S1 {3 R

OLECHECK(this->AddInt(x, y, &ret));

0 M& m- K$ w0 ~5 u' x

return ret;

2 C( Q; S. S( i" B R7 |. ?5 j

}

; ]( Y. b$ |2 E; u& R* p3 _

#endif // __TLB_NO_INTERFACE_WRAPPERS

/ d$ F. k9 y& }/ d: x6 ^

};

8 w+ I' n+ k) g4 H3 O7 p$ m5 ?1 R6 S

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

/ X) u4 P& ?- f! q0 l

template <class T /* IMyCOM */ >

9 T0 F( [2 D. U, v @

# D3 w s& n5 \

class TCOMIMyCOMT : public TComInterface<IMyCOM>, public TComInterfaceBase<IUnknown>

% \2 T M8 @, U4 m# R

{

0 Q5 a7 \1 d5 s/ j! W# `3 Q

public:

/ L! | I4 e- n: e

TCOMIMyCOMT() {}

- Y+ d2 D8 v; m+ x

TCOMIMyCOMT(IMyCOM *intf, bool addRef = false) : TComInterface<IMyCOM>(intf, addRef) {}

7 ], X% Q* Y) p7 \

TCOMIMyCOMT(const TCOMIMyCOMT& src) : TComInterface<IMyCOM>(src) {}

1 @; c" L0 D( q8 |' l$ A' n

TCOMIMyCOMT& operator=(const TCOMIMyCOMT& src) { Bind(src, true); return *this;}

9 {; v: f# [3 d/ H# n& G. s

HRESULT __fastcall AddInt(int x/*[in]*/, int y/*[in]*/, int* ret/*[out,retval]*/);

2 g/ T( \6 [% e; r" E

int __fastcall AddInt(int x/*[in]*/, int y/*[in]*/);

" E+ C' v# ~ d) k$ k6 N) V

};

h# Y+ p' s3 j5 t" _

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

6 N8 E7 k: g2 N# K4 I# A/ O8 I

% Z( O. Y$ E: D5 u6 J

template<class T>

! Y5 N1 Q' i; A2 S) d

class IMyCOMDispT : public TAutoDriver<IMyCOM>

8 @! o# L F/ G+ L# n1 K4 I' ]

{

" J( u& |) g l4 _ Y! x

public:

* c# b! @5 x) C

IMyCOMDispT(){}

- |7 ?/ ^1 v! N& r% P4 J

IMyCOMDispT(IMyCOM *pintf)

: r& X( n8 J" W* R& x* L, q# D+ B

{

( n* @8 x5 {2 x: F

TAutoDriver<IMyCOM>::Bind(pintf);

7 a: B4 f7 {# w, B- K: e

}

1 ?5 u7 }8 k9 n" i

IMyCOMDispT& operator=(IMyCOM *pintf)

& i) L( F$ Y# W

{

4 R0 X1 z& p3 z) B, E

TAutoDriver<IMyCOM>::Bind(pintf);

. b5 L% s5 P# J% \; h0 b! V0 ]

return *this;

0 {2 q$ O/ m% r2 ~7 a8 N( F5 w

}

1 @+ @" P* e' c1 m5 W

HRESULT BindDefault(/*Binds to new instance of CoClass MyCOM*/)

0 U1 Z% Q) T9 y3 ?' I) o

{

+ Z4 y- D* z3 p8 n, J/ y; v

return OLECHECK(Bind(CLSID_MyCOM));

$ @$ E: {8 w2 `& u8 `1 c

}

% f: N E( s! P7 E! d, L1 E

HRESULT BindRunning(/*Binds to a running instance of CoClass MyCOM*/)

4 X2 T! Q; R0 I( H" a- e* e

{

9 \) N/ S$ C# ~0 p6 {. c1 p m; {

return BindToActive(CLSID_MyCOM);

3 G# }, S- `7 ]: }% N- j

}

, d$ X1 y$ {! I5 A! Y

HRESULT __fastcall AddInt(int x/*[in]*/, int y/*[in]*/, int* ret/*[out,retval]*/);

$ n* w% ?* e& k$ c) B

int __fastcall AddInt(int x/*[in]*/, int y/*[in]*/);

7 T+ v a+ T* X

};

& P4 U2 u3 J: G; A3 S2 C) p& ]

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

3 k4 S" i! W1 P$ b- E" C" Q. D

# J) u9 f) P. Y" q! K: d' h$ s

template <class T> HRESULT __fastcall

. ~) d* E; N" |5 p: @; w2 Z

TCOMIMyCOMT<T>::AddInt(int x/*[in]*/, int y/*[in]*/, int* ret/*[out,retval]*/)

6 B" m! H# S, h1 i$ w) b. A+ t

{

! M5 u3 z+ ^' y3 u

return (*this)->AddInt(x, y, ret);

! ?* P7 y! y% q( f( X1 c9 c( T8 q

}

' b- U5 A- E9 @) A

* p# d" l7 E: b f+ U. T

template <class T> int __fastcall

4 x* W5 _3 w* C. A" R

TCOMIMyCOMT<T>::AddInt(int x/*[in]*/, int y/*[in]*/)

/ K& f8 D d' R; F; Q& |# i

{

2 M1 l. i5 D7 ?+ j( Y- A

int ret;

0 T6 M2 u' M3 ]$ q$ W

OLECHECK(this->AddInt(x, y, &ret));

b# H" l3 B: A# D" N. S" u7 Q

return ret;

& i+ s3 c8 |8 D0 o

}

( g' W, j! F: [& G) n% u6 c; |

4 ?5 @5 G4 e, L3 i

template <class T> HRESULT __fastcall

4 ^* ?' R; b9 Y1 |. i3 m

IMyCOMDispT<T>::AddInt(int x/*[in]*/, int y/*[in]*/, int* ret/*[out,retval]*/)

% @- x2 R5 \9 K% W

{

. C( R& T+ L1 |# ?; R

static _TDispID _dispid(*this, OLETEXT("AddInt"), DISPID(1));

# P) i0 A! s- c) N9 g9 `

TAutoArgs<2> _args;

% p6 u5 w' v7 K! N: e4 b

_args[1] = x /*[VT_INT:0]*/;

( p4 B" i/ T; G8 I6 D2 U

_args[2] = y /*[VT_INT:0]*/;

" }1 D7 G c4 y4 Q$ g2 v4 f

return OutRetValSetterPtr(ret /*[VT_INT:1]*/, _args, OleFunction(_dispid, _args));

' C' h7 c' r7 d5 y* v. O

}

. B& o/ @, d. z ?( q0 w

- K5 r* F9 C" a) C$ v X9 r9 f* O

template <class T> int __fastcall

: s. G& a- D) H3 l5 i

IMyCOMDispT<T>::AddInt(int x/*[in]*/, int y/*[in]*/)

& S! M& y- c, Q( U

{

" k* i( g( q: i' H) N, [5 s

int ret;

: g% k! C+ E1 L3 B+ L

this->AddInt(x, y, &ret);

$ J0 ] j' `# J

return ret;

8 c0 y$ L8 w/ e2 z h. S

}

1 U' Y6 h' a Y9 n' }. X

) G( i+ v/ e) P j& S/ w

typedef TCoClassCreatorT<TCOMIMyCOM, IMyCOM, &CLSID_MyCOM, &IID_IMyCOM> CoMyCOM;

! r Z1 M# N9 ~# e$ k8 P

#endif // __TLB_NO_INTERFACE_WRAPPERS

3 m6 K2 ]7 a0 Z. Y

}; // namespace Pcomserver_tlb

5 f% l( R+ ?! n, B- f

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

1 K {" F( `/ l6 q2 I E& h- D

using namespace Pcomserver_tlb;

& W0 |* W2 H9 O, G0 |

#endif

" u( m. D2 }3 l* B

#pragma option pop

3 Y3 z; r& N% d# y

#endif // __PCOMServer_TLB_h__

( c, v3 @: }" W1 O5 U0 f

下面是文件中说明的主要对象及其定义:

n# v1 @) C6 ~8 G( g, S4 S1 y& ~

interface IMyCOM : public IDispatch

* r6 \! ^2 v% \4 T6 G" F; B, B

class TCOMIMyCOMT : public TComInterface<IMyCOM>

X) D) E; ^. @ D) X Q! j9 `

class IMyCOMDispT : public TAutoDriver<IMyCOM>

: b# H% _0 X) g$ i# m k

class CoMyCOM: public CoClassCreator

% F/ ?2 W, ^4 V& u

◆IMyCOM:通过IDispatch接口来调用对象的方法,该接口可以使对象被不支持虚拟函数表(VTable)的语言(如Visual Basic)说调用。这是一种很慢很苯的接口调用方式。

/ r/ u! S) b; B' K8 v

◆TCOMIMyCOMT:通过所谓的智能接口来调用对象的方法,既可以实现IDispatch调用,也可以采用VTable进行调用,从而实现最快的调用速度。

$ f- Z; F \% v. q7 M

◆IMyCOMDispT:通过disp接口来调用对象的方法,可以提高Idispatch接口的访问速度,但还是比不上VTable接口。

$ i9 O! v6 e2 | V+ G# L' F

◆CoMyCOM:通过使用CoClassCreator可以自动产生TCOMIMyCOM代理的实例。

& d7 k- S) k# T9 _

下面介绍一下实现智能接口和Disp接口调用的客户程序。这个客户程序很简单,有两个按钮分别完成两种接口调用的方法,一个编辑框显示结果。

, F. d$ \6 D/ A1 X# M; x" t

◆智能接口的VTable调用方法如下:

/ g4 [! }; Z) x+ D+ P

int x=6,y=6;

1 v* X2 y7 D9 P6 A% A8 ]. _

TCOMIMyCOM O;

# ?* ?* ^4 N L7 }$ ?+ `

O=CoMyCOM::Create(); //通过CoClassCreator完成初始化

/ S3 l2 G' c+ v2 @

O->AddInt(x,y,&y); //Vtable形式调用

! p8 q# @* q6 L ~0 v" m2 J6 l

Edit1->Text=IntToStr(y);

( G2 c% |8 s* Q0 ?' R+ \9 C6 ~

◆DISP接口的调用方法如下:

( A' H; U/ e( o7 r T( o U! W: f

int x=6,y=6;

/ @( |: z( x' [3 o# D

IMyCOMDisp app;

& S- A6 ]$ ?. Q5 e

app.BindDefault(); //通过Bind完成初始化

; m+ b* h" U2 W$ i

app.AddInt(x,y,&y); //Disp形式调用

& p) b( o# }; [+ q( i

Edit1->Text=IntToStr(y);

7 {+ F% q: E) G% W! U6 l

4.小结

: q# E' D, c* @* T3 I9 V

上面的程序举例是很简单的,但却详细说明了COM应用软件的开发过程。COM技术不是一个编程语言,而是一种编程规范和方法。采用COM技术可以开发功能强大的软件,有利于分布式应用技术的实现,有利于多人合作开发,也可以帮助我们理解Windows系统本身。COM的接口技术是比较复杂的,想进一步了解COM技术可参阅清华大学出版社的《COM技术内幕》一书。

3 F3 G4 q* w D- J- P

C++ Builder是开发COM应用软件的好工具,它隐含了COM实现的细节,我们只需与它打交道就可以开发完善和强大的COM应用程序。希望有更多的人转到COM应用软件的开发上来,COM技术是软件技术未来的发展方向,是实现软件工程中软件即插即用的有效途径。

7 w. D% ^3 h; s% l; w. b; e" O

& N8 k$ \3 _$ J' |- q! |

zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2026-4-20 12:17 , Processed in 0.431225 second(s), 52 queries .

回顶部