数学建模社区-数学中国

标题: COM应用软件开发技术 [打印本页]

作者: 韩冰    时间: 2005-1-26 01:07
标题: COM应用软件开发技术

蔡倩

& e# h u8 b8 X) l3 ~2 Y

主题词: COM ActiveX C++ Builder

' \; {1 c0 y2 x4 `" o" ?' E

1.COM技术概述

7 G) ^) E" w# w

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

8 D# @6 b% ?+ g j) t0 H

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

* D1 S( H0 r% g& s

◆用于创建ActiveX控件。

( R2 s0 a& S) ~; N8 o7 e5 U

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

& X' n9 S( I% H% I( s+ ]

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

& V% @+ q. \ |, N

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

# @2 @ U9 ~6 ]. r; {+ }

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

+ E& M3 ?; q1 ^4 I A

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

3 M/ Q, \: J; t: @: Z: f9 Y; u

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

7 g e1 B4 R) |' [2 g4 ]

2.建立COM服务程序

9 v5 N# G( `: S( a* \

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

0 W& b6 u; F: Z- z# V2 Q, g5 p

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

1 F; k& p% O2 R) a+ Y" C' K0 f0 M

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

" O0 }+ z, h$ I: \/ a

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

# Y4 L9 x' O, W, E- G% s8 r1 G

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

$ s) F |, A8 r, q

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

3 k3 T7 ?5 \" u8 Y1 B

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

0 l9 U+ D& ]& ]3 ~& w, p# n

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

! E9 Q9 w$ W$ I

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

9 i9 A! ?% o* L2 J" ?9 t

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

( n; R- s7 y0 Y1 H& m

2.2建立COM接口对象

% d: K8 Y9 o: ^) m

1 O" N$ H/ @4 l6 d1 ^3 {

+ O: W6 h# b0 c5 e$ b! ~

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

2 | z# w7 Y4 l

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

7 m! A) \2 d( K0 \. e8 O) d6 {

7 z( {+ q5 Z- A) A% E

2 ^& }* L! z; u2 P% {1 I1 W

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

- ^" c, s/ P4 X

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

3 t8 `- o) K5 i- O8 \! M, ]/ ?$ ]

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

" }. O. a+ s" ^6 `' L

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

3 C+ x+ x4 R3 `5 H

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

0 u" k, @* K& J5 R

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

% F* z- v3 Y* L9 g5 Q

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

1 t" y; L! j1 P( M, Q; M

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

$ N5 b( ^. V5 K

[id(0x00000001)]

5 a6 K3 G& |" x

) o9 I6 G& R' J

- b4 U3 |* d/ ~

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

( ?# t4 w; X* R5 o( Q

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

0 `1 N" L* |0 |

PCOMServer.TLB: 类型库文件;

( d& l% r# g( k; E

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

) O& @% W* n$ F* e: b

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

( g9 w t! P% S/ U' Z

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

: }$ l$ k# a9 C( x' `

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

6 F. n+ L D" y) t0 e% L

2.4 实现COM接口中的方法

8 g( h, a9 B: o3 d2 Y

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

/ [% w, t9 i% I( N! z4 t

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

6 N- B) U) _' [, n" x

{

) E2 L& p+ q( q" ?, i8 S" I, M$ G& Y

*ret=x+y;

% W7 f m7 _ J

return S_OK;

, c F* |; }3 S' a) N2 T

}

2 _" X3 i2 s2 k7 H0 `- W

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

8 J G( r. |) g$ c# W Q

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

2 X) x4 h S. s M* S- k6 T' y ~3 S

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

) P: N4 J5 z) V- ]* Z4 P! n

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

& ~/ |7 \# z/ }) ?6 ~8 V

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

- n$ t/ P5 C* G8 ^

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

; X4 R5 B# T7 i, ~: ?

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

8 D8 j1 K7 m* } c* O* B

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

, ^3 E( a9 o1 m6 W u3 k

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

; I, t1 }( g! E, X% x7 L+ G: k

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

3 S+ e9 H- _* G( g# q4 N) L

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

( j7 O' w5 n3 a Y5 U

3.建立COM客户程序

) c& x, G# ? f0 M6 P! p

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

7 z0 I6 c' | [$ P) o' |

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

: E, E3 V3 I0 t" w$ K! `; `

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

) w9 U# x D' ]0 O0 }+ f

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

5 O, [, L7 z. ]0 P

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

3 g" \4 A0 a% M3 v: |0 Q

// Helpfile:

/ S( P. q6 L8 @& z% b

// DepndLst:

* k# h& l1 \7 q( r+ o) v

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

2 e% P6 }: [5 l+ t4 {

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

4 z' [: V! I& \+ Q. u8 z0 z# g

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

( h" T( `+ T. H: C9 z f

#ifndef __PCOMServer_TLB_h__

! d* N6 `1 `4 c& }5 X* _8 q

#define __PCOMServer_TLB_h__

7 |* h& B: e4 g4 |# ^

#pragma option push -b -w-inl

% U z. Z4 \& _) n

#include <vcl/utilcls.h>

, V: Z& \3 q7 y# \; T

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

& B2 y' g5 c; p3 D' U7 p. m

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

p/ L0 W; @+ |. i D. F; L: n: W

#endif

1 v: R9 L; y6 b& Q

; {# M. [/ ]+ q

#include <olectl.h>

: x; Y* b0 g. q& B0 d& l

#include <ocidl.h>

8 G6 a0 Z& T( p; R: p

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

6 ]* A8 }: C& i* H- A' n- E* J# C

#if !defined(__TLB_NO_EVENT_WRAPPERS)

7 _+ W' @, ]# h! ]) R

#include <atl/atlmod.h>

+ v3 I6 m$ j( _, }3 v3 s. D

#endif

1 d$ ]- }: q/ a0 y" M2 ]" u

#endif

; t5 \ s+ r$ e8 v# q" i) {

. l. [( n5 u2 O) d% l @

namespace Stdvcl {class IStrings; class IStringsDisp;}

" D, W" `. ]- V: W

using namespace Stdvcl;

5 f& ^* Y' p& }; b1 J6 M1 I9 w4 ^

1 e5 X* L) C( }$ i% H; t

namespace Pcomserver_tlb

! w* z( e9 E5 k8 z4 V; `. k

{

: {0 h7 L/ b$ c6 @+ a Q

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

9 _2 e& h: Y5 \

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

: Y, h5 T+ N' C4 T3 G5 }0 [( y

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

0 q7 X. S/ t+ o2 B% H( m1 P

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

% J* d. B- d3 l+ A: X) `# R

typedef IMyCOM MyCOM;

; C( D6 ~# O4 Y% g1 @$ P, t+ K" J; t

5 H8 U, S: q/ R( E W

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

/ A% l/ o; ]- J. y5 {

interface IMyCOM : public IDispatch

3 v8 q6 ? y1 q: [8 W; g: |( y4 @

{

1 G( ?( H# a% |, W

public:

3 t2 F1 c8 M. X2 W5 V, n: F* ^3 Q

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

7 O0 a8 g3 S! P6 V4 e

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

+ T7 l l ~' O; K+ S3 `/ k

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

J- D/ H% j0 F ~0 z# ^

{

% L6 k4 a7 u9 X2 O7 k

int ret;

; N: c1 G" Z1 E& r+ T

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

! S% H; O5 M! G9 B

return ret;

* x5 g j3 S( W! Q, E7 r

}

" w. K5 ]/ G8 `9 L3 v _

#endif // __TLB_NO_INTERFACE_WRAPPERS

' y3 e0 X( w' T% C6 m

};

9 S' z3 z5 W( G/ g7 f6 L8 @) ?

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

! T% V' k0 S7 q1 ~; w9 b

template <class T /* IMyCOM */ >

% m# A4 {8 V# ?$ T( m

9 {+ X3 s5 s5 z- c! a

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

7 |: Q! J$ L: q3 z

{

1 L2 H* c @. \1 t* N

public:

8 q, D8 Y8 N6 R: e% \, H4 b

TCOMIMyCOMT() {}

2 [. U/ x# w! S; j4 z- B$ X9 U

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

; _/ @3 P$ u& V' {

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

8 u) w* c& o5 R: @

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

: a% \& b. {3 p+ r( V/ t0 I

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

+ \6 c: o2 `/ k

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

- J5 @5 ~5 T1 W) l+ W

};

! y- {/ f$ S& r d p+ c

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

: J1 @ a& p& k G n

9 ^2 r. W) [( R6 U

template<class T>

2 `4 @2 e. c2 F2 I; @: X

class IMyCOMDispT : public TAutoDriver<IMyCOM>

: E2 p" a0 q6 z8 A8 b5 e

{

5 q0 ?0 C7 u2 V, w

public:

7 N6 n6 n5 {/ T& Q' m3 Q. L

IMyCOMDispT(){}

/ f+ t0 d, U& \/ K8 S& B5 h( m

IMyCOMDispT(IMyCOM *pintf)

, }& _: t. D8 s6 w1 c# a

{

$ U o) @- P( ^4 X' H" w

TAutoDriver<IMyCOM>::Bind(pintf);

- m' F( o b, e% s; A: V% k

}

; w8 a4 Y: [2 o6 Y! {

IMyCOMDispT& operator=(IMyCOM *pintf)

+ p4 g; i9 E: k8 ~& P6 G. D) i/ [

{

5 C3 T! p8 y& X9 b8 D, y

TAutoDriver<IMyCOM>::Bind(pintf);

( e8 |" f( j/ q4 Q7 V6 y m" Z

return *this;

- @# M: Z, _/ B& T, D4 Q7 b5 T

}

6 ~7 }$ i/ h; @& w U

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

8 ?* J: O1 Y7 O7 G* Z; [

{

; W& b, B5 D! M

return OLECHECK(Bind(CLSID_MyCOM));

: |+ L# ]9 R, O6 j' z3 B' z& i. \

}

0 v, M) W ?/ X7 ]8 p5 l1 {

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

, N- k9 w( U- Y5 D2 I! Y

{

3 ^2 q, M; B% P- x. m1 ~: V

return BindToActive(CLSID_MyCOM);

m% { K! H/ i. q

}

" D) O. ~' v: O+ ` {

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

3 a# D6 b& Y: ] i

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

% S d0 F1 f- \# t5 W

};

( M+ B, Q5 G7 k" h

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

8 L: S! ~6 o( [& Z

: V G( i* p; a0 {0 U! S. P( f

template <class T> HRESULT __fastcall

4 ]6 j, t& S/ b R6 B

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

0 b6 o3 \! D" k7 N8 J

{

) I& d" P0 l! _7 l# g

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

% S+ u$ s) e" H0 }

}

4 f5 u1 m& i' I, A4 ^* w+ r. B" }. h, f

% e" `, W, I3 w% Z* O6 T

template <class T> int __fastcall

|/ q0 D* |! G: a6 h; {5 n3 {

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

. |1 Q9 ?) C. {6 {# Y

{

" X/ P1 q9 ?, q) f* Z l; B. ^

int ret;

" L2 |; ]# z2 v$ G& _: k4 T/ }* g

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

/ f7 g0 y$ A# [- ?7 ^5 b( T

return ret;

% H2 ?5 B; V: Z. w* o9 M

}

! z8 O. b9 \. j D! ^6 O

5 W* [" P6 A, z1 A% n! |+ t

template <class T> HRESULT __fastcall

1 X/ S$ w+ M" c7 F/ r

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

- F& i$ } `1 @: K& Y

{

9 m0 f4 w0 U3 J8 u

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

% M" v- ~' j5 b- V2 W6 k

TAutoArgs<2> _args;

. o! r+ N1 f$ t

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

+ M% b$ Q5 m2 f7 r1 |$ N

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

8 t7 m2 I9 a- U1 o+ u1 b* c+ `

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

8 X: e$ v6 V9 H. B9 q" U: s) L- n

}

4 k- _: W$ u+ l2 H3 j

/ O$ z. c# Y, Z" F) q6 A# Z0 N6 ^0 _& i H

template <class T> int __fastcall

4 H( H) N; h8 v) O

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

# [3 ?# l/ ]( c4 Z2 t5 M. G7 f

{

) a0 o; U9 F3 a x6 h( M5 c

int ret;

) c0 y, i2 x$ D2 E G! ~

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

: N" W% ]# h$ t9 S& i

return ret;

) j/ g6 J7 c& l

}

2 E0 O7 }: J; k+ w* ~% c+ I

' o: w, H+ p" N$ `

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

9 Y: T, c; n, m

#endif // __TLB_NO_INTERFACE_WRAPPERS

" o9 |2 Z8 m" s3 M& e2 R+ R

}; // namespace Pcomserver_tlb

K5 b+ K( ^1 Q& J

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

* ^8 D. ~0 I* V" L

using namespace Pcomserver_tlb;

, Y! p) M6 k H# }3 Q% l5 G2 x

#endif

) V6 j" q+ d F4 Z

#pragma option pop

( j& O* q9 X1 A9 b8 V( s" d2 f! |% B

#endif // __PCOMServer_TLB_h__

8 `2 Z2 o2 z6 i! d. o! X

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

# W7 o3 v1 l; ?2 o+ r2 v

interface IMyCOM : public IDispatch

3 |$ J4 r3 f: A; H6 s. X) D

class TCOMIMyCOMT : public TComInterface<IMyCOM>

/ ]1 ?! p: h4 W/ E% e j: }) P

class IMyCOMDispT : public TAutoDriver<IMyCOM>

+ t7 q. D4 `. S# x( F8 E

class CoMyCOM: public CoClassCreator

4 u7 M" |( Y6 a9 I

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

% x- s3 ], ~ R

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

% Z1 {/ u- e; }; t; X

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

+ |% i9 J3 _9 _+ T- n* k

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

: f- Q$ b9 ]. ~4 {0 Z/ a, I

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

+ x& j7 k+ C3 C- \( {& v4 F

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

6 ~0 a! y% R/ [7 e7 {- e! K

int x=6,y=6;

2 L& c' _( M2 Z# |& d: b

TCOMIMyCOM O;

' c* {# P1 [5 S9 z

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

* N6 f/ D+ \2 |4 N# I: w5 B% g

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

: Z* X* C2 Y% O1 p0 H7 G

Edit1->Text=IntToStr(y);

4 X3 q1 D8 V: @! |4 K# _

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

. h' e+ U2 U- Y! f8 _1 Q

int x=6,y=6;

x Z6 O' M3 F# |6 u$ E

IMyCOMDisp app;

4 r E4 B) K- z/ `2 D

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

9 m y4 P% Z' c- m. v2 n

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

- y, `8 Z; F: C

Edit1->Text=IntToStr(y);

" j) L6 B; u9 A% k

4.小结

" Q$ O( o; f* ?% v% R

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

, x+ C- }" J5 `2 |

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

5 ~: w( h' L0 ^6 \$ J) m. P

. g) p/ j$ z1 T" }0 {






欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5