QQ登录

只需要一步,快速开始

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

COM应用软件开发技术

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

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

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

蔡倩

5 g5 U ] j, X# G8 R/ }$ C

主题词: COM ActiveX C++ Builder

2 Y$ [ O6 W U* i! h: [2 b0 V. B

1.COM技术概述

' _! J. M7 ?5 y0 N u' a5 B

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

a- [. |9 W; H t. @! k

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

2 e( t9 X+ H& W, c

◆用于创建ActiveX控件。

& ?0 J6 j0 R0 p: m: z2 D: P4 v( x

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

6 r+ y. r6 o4 T5 m

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

! j; ?" G) ?* a

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

( X# ?# L1 a) x; `6 I

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

. f9 M- \1 M# {* n

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

1 C; N/ x! [$ e3 q1 t

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

5 E' Y6 o( Z- S! W9 O: F

2.建立COM服务程序

6 z2 D" Y0 O) \3 R7 h

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

; {+ y" x6 \! I0 P( b5 T

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

, Q2 | v0 X7 ~' p' i- o% \4 L! s

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

9 W" R5 l5 ]/ y; x' Y. k8 B

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

) q7 i( J9 Q, Y" d

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

) o! I! p e% R0 R" {

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

- r. w7 [/ Q6 B6 Y6 Y7 w! m% t p

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

" F, s9 k- j; d& b+ j3 _

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

+ P. [2 g- @5 v" \ o4 u

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

2 C5 f, j* h4 M- o' o/ \ X

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

" a! n) t( Q' n; c8 C1 R9 s0 g3 p

2.2建立COM接口对象

0 v1 ]) `' q" g6 U

$ ^4 I) V5 V! i

# Q5 }8 n7 v! _* F* B

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

- B# B: p. b7 g1 k' k ?

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

1 ^( V( h+ U4 Y; c! o3 m

3 V. t$ b+ p6 z- [$ A6 G5 M* B

! \2 T g! R+ ]3 F% l. s: E0 `9 g

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

5 j( W+ o$ J# X+ S5 D4 F2 E* w

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

) K2 Y$ _# h9 L% m* p( A

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

- E1 Z. B% }' d1 n# O3 U

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

" y6 B6 q9 @6 o T

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

9 C3 P9 c4 C' P' P6 N* i( Z

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

, J- z: g5 j! I1 K2 q R" u

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

, L7 W0 N* F; S9 P3 e) h) k1 i1 q* ~

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

! k3 {4 \. ]0 o; O. t3 [

[id(0x00000001)]

; Q# ^" E z6 d" g# J1 r

. M; F5 r; }7 K: ?6 p2 `

5 [0 ]" ` q( ~ t) s

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

. V/ |" j0 M! G5 ?: l

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

" t% W, ]2 X. m( W' M) T3 E6 H

PCOMServer.TLB: 类型库文件;

5 A# G$ O1 k& d3 E1 T5 G: r, F

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

( q0 x% ?/ A$ o& }

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

C0 W% @( c8 a8 Q7 u

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

8 B( A! e8 ~. n2 F# a4 u' j

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

$ k" J8 ^' k, x6 E, n8 z

2.4 实现COM接口中的方法

6 i& b; @6 T' g \* u

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

4 a4 X% C |) d

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

* L# s8 g) z2 r0 E: K

{

8 L/ `. m) J( h4 Q0 w! g

*ret=x+y;

6 ~' m6 ?+ H) I0 b9 i8 H+ S- |

return S_OK;

. |# w0 l$ q4 H4 n1 i- L

}

4 U: Y/ p1 p0 J) s2 a1 K

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

/ g; i( l# {" ~- i: _ h* s

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

( ^ f: n3 j9 I% W( N: d* ^1 r( ?+ f

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

/ B9 y. R$ n2 T$ r# A

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

: i. P% p; K9 U9 \

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

4 r- F" U! `; d5 t; R& R, Q

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

p6 A) H9 A6 M' K8 M4 |

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

' w) z* b8 @1 E. a+ S2 @

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

9 S" k& ^9 b M5 O$ l# V6 ~' N- ^* i

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

- t5 g- R# P4 j4 j

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

- E' L+ Y; l. }, |& n

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

! [% r) N6 D) B0 E# C2 G* }

3.建立COM客户程序

) v6 I+ L+ g' C \& _; H1 y7 _

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

4 p7 S7 {# i2 S5 R

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

6 r& R" y) U0 p- {2 ~, }" u' @

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

0 J$ M0 Z5 D! c

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

& Y! A! M2 d* ?4 H6 M5 N3 m3 A

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

$ l" L, v( g4 O' `

// Helpfile:

% P8 F( l# @, Q! L2 a1 c

// DepndLst:

* C( Q) L$ X$ @+ ~4 N+ `

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

8 _; i# N7 v S3 F

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

0 X( y5 ?/ S; l% Q8 E

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

$ E& L; d- n5 z" K/ ~4 ?

#ifndef __PCOMServer_TLB_h__

# F) p- e0 { ? I9 A9 o* Q

#define __PCOMServer_TLB_h__

7 C4 s# q, f n9 d& x2 I9 S

#pragma option push -b -w-inl

* L0 e+ e$ a% C5 l+ E

#include <vcl/utilcls.h>

- U4 u* P# [3 K. j4 y2 c- ]

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

4 f, R4 w2 h. l' G5 [

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

6 y0 J: O$ m1 `( f

#endif

y2 X B! H6 {8 n% ]/ B* F7 [

* m: S. _8 i6 U; d: E. g2 _

#include <olectl.h>

9 r* }7 d0 g7 C$ e$ w3 o, R

#include <ocidl.h>

. j9 L/ ]* Z `" k! u) \

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

* L4 ?, @, ]3 [' q" p

#if !defined(__TLB_NO_EVENT_WRAPPERS)

& b& D" g. |& Q% M- d z" q

#include <atl/atlmod.h>

8 Q( V- n9 q* u, \8 c

#endif

) m7 P' E* ]" K" J# T

#endif

7 {. T& W8 g+ w5 i# i) ~$ O

]+ O j8 ]/ M$ c

namespace Stdvcl {class IStrings; class IStringsDisp;}

0 r2 o5 d) O x$ M2 m, v7 N+ r

using namespace Stdvcl;

; D! d9 [( l! s! w" E, o

' U1 Z1 h2 E* n9 p. @1 v7 L

namespace Pcomserver_tlb

W0 J0 {9 A- {8 y

{

2 a0 @& {$ N8 W0 c, S9 i7 S

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

/ `! }! D$ V& W9 Y, |5 Y# p

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

- P/ p( _) ?2 ~- J; h2 F

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

$ X# R5 g( `1 {: r+ P# F$ ]

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

0 X1 H9 J3 E) {$ v9 L! X+ w3 k

typedef IMyCOM MyCOM;

+ y! e R O8 @8 E3 N

7 P t1 Q! n# G

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

& S" O. d$ \" f! W3 V, A

interface IMyCOM : public IDispatch

) {7 x4 \6 F# n

{

, v9 d) q3 X* V! [: V7 A

public:

( n. o+ t C- q6 W

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

. h- a- s& D) l, W

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

6 y( s$ a) p i% h2 F

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

* ] ^, a, {( M' \

{

+ h3 q$ Y0 u& k5 \

int ret;

& H0 [5 R! ~3 b- Y0 d1 ]& t" |( B

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

. k* |/ U5 W( y! m f9 S! }

return ret;

4 S6 M+ o7 }4 u; ?+ S2 t

}

. r5 g: G- g" N

#endif // __TLB_NO_INTERFACE_WRAPPERS

* A7 A+ Q3 Z' q* g' }

};

( A( e: d! Y; P

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

4 f, j" ~4 O$ Z' I

template <class T /* IMyCOM */ >

. J1 o. ^ y2 u1 q6 I Q6 X7 V# f1 C" o

% ?9 e1 h+ b) x6 `3 P5 E

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

+ W) K1 K# R& H2 \( U& N# I

{

8 |; [% ]6 M E u9 U

public:

' L. y6 d, J! y- y. m, t# q

TCOMIMyCOMT() {}

9 N% j, c) |* d9 h7 B( d& q

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

* n# J& g: f" i+ U; T

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

4 O0 G; R9 G b* h

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

/ }# }; v2 ]# c. ~8 f4 q

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

! K6 M( r8 D" p0 T: c' h; X

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

( [. ?" M* S$ F- P" H& _0 h( g

};

: T* ~1 `5 `( A- @! _- o E

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

6 v( B9 g+ n2 y+ P( z, Q

% M0 A1 w5 r5 U" C2 b

template<class T>

2 n' c3 U: y- }) L+ Z) D

class IMyCOMDispT : public TAutoDriver<IMyCOM>

8 W' R( H+ p: c1 o9 J: P H+ i3 G

{

' p4 o/ L! ?9 O" U

public:

: R/ q F8 p" \) t) J" _

IMyCOMDispT(){}

0 j* s% R3 @" |& o' O: @ a

IMyCOMDispT(IMyCOM *pintf)

( r( ]# B+ y& X( s- [

{

1 `% U7 e' u# {6 J' n/ P% i; d6 I

TAutoDriver<IMyCOM>::Bind(pintf);

6 d' w9 ~$ E& S1 O' V; h; W

}

6 I {' `$ @' z n

IMyCOMDispT& operator=(IMyCOM *pintf)

* ^8 U& q/ p4 m% r6 w4 x) y0 O

{

( H7 G5 I1 R& D/ y

TAutoDriver<IMyCOM>::Bind(pintf);

! m0 O* |, B" a

return *this;

9 n" |4 @5 J9 ~( F$ s/ L! ~

}

2 v0 }8 E! B$ z4 g

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

6 j, t4 S% V$ D

{

1 w1 T% E5 I# T( k! W3 \8 b7 m

return OLECHECK(Bind(CLSID_MyCOM));

& {' I- ]9 b9 L; K5 ^. d/ E! W

}

& C, g- ?2 Y' m

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

; C t- f# F7 q0 Z

{

1 @ q' A i* K. E1 a/ N' a

return BindToActive(CLSID_MyCOM);

6 b/ q+ x8 I3 K9 m$ }

}

; p8 F I6 E5 m! t

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

7 b4 X8 s4 ~6 u: s

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

3 u+ x/ U& N5 P, b7 Z

};

3 D7 J9 D0 T" C: _

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

+ {/ U; j- E3 T! c

$ w0 L9 i& K* R/ Z' N

template <class T> HRESULT __fastcall

! s9 ^# `" Y J5 y8 X: k% u5 @9 w/ K

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

1 A9 P3 ~& }! J

{

# h3 W$ M" X" e1 G0 v

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

! `8 D* ~. |. Y* j

}

' a! N; ~+ ~2 _- C- E% r s, X( [

0 O! x# S" l. [/ G- |) b- y3 k* \! N

template <class T> int __fastcall

8 R0 x- @# I3 s

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

- y+ D& g* b4 ^+ H% w/ T

{

. F4 ?( `% t6 `7 i* Q+ a

int ret;

4 X5 D# `# u9 j& Z$ B' ^" g

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

; G. H* N/ G0 ?

return ret;

4 ^* {4 \ G" C+ y' f6 e4 ?! M

}

' E# X4 i2 l: N. f% T

# _ \$ }- }# {4 L" a6 s6 }+ T9 W

template <class T> HRESULT __fastcall

* u; g% E4 C7 W Z4 X

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

- I! o/ c9 \) w; S7 ?- `7 W

{

9 Z6 A+ E/ k+ S( v/ Y/ }- y" \9 w

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

d$ g1 L2 X |9 ]1 W! ^

TAutoArgs<2> _args;

) Q& H* y0 |4 |0 L

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

* p$ p% ^- q( I3 s$ a( O+ n) {( e

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

, n Z6 t5 z# l' a

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

0 C& W% k5 L4 h

}

8 i! t( q. R7 U @! C& M+ \

' Y. J! W k% L d# l& k

template <class T> int __fastcall

2 Q& ?' D1 _% G* o8 H0 e" q+ j

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

$ y3 ^- y; W& j3 T6 _

{

* w- s6 H: e+ G7 k2 a9 G# R

int ret;

$ N/ B) X" W( P1 }! t

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

& K; v9 S) \& l: m5 F a

return ret;

% [% X# U2 y% f* q

}

, J2 _4 x6 r, c: E

0 `9 H4 e5 e4 c, l8 i8 G

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

L3 X2 C( ^7 F) y

#endif // __TLB_NO_INTERFACE_WRAPPERS

$ o9 F) v% ]$ _5 I7 h9 @

}; // namespace Pcomserver_tlb

- x2 {' a, G& g9 D, Q* d# F

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

+ M: F5 W' ?& \" a5 u

using namespace Pcomserver_tlb;

' o: a: Q4 J! r+ X* i) {

#endif

) q$ }- Z7 v6 M2 j% H2 c8 u

#pragma option pop

6 h4 k3 K' ~6 J: o8 t L" y; R

#endif // __PCOMServer_TLB_h__

% m. o+ z( y8 h% W; d* h9 X

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

, c, |+ v( t3 N

interface IMyCOM : public IDispatch

, c$ V) r( v: `7 v0 f) q& p

class TCOMIMyCOMT : public TComInterface<IMyCOM>

5 m& |7 Y: l0 A4 ]1 j, N( E

class IMyCOMDispT : public TAutoDriver<IMyCOM>

4 O: D. Y0 W! j; ]& G

class CoMyCOM: public CoClassCreator

! \& S: M+ x' S# H

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

$ `, k* {$ d5 ?2 G

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

9 T7 }. q7 G: j5 x E2 Q5 G# H7 }

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

! b( b/ f& }) f# f

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

0 }+ f; [: R! R; P' `* p

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

& l# V4 U4 p0 {5 s* ?- ^# Q

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

0 B9 \: ^' G8 y' T+ m

int x=6,y=6;

$ f n3 c8 O3 g+ F% k. L. w. g

TCOMIMyCOM O;

: x: l/ y2 v2 H

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

- E1 {5 x/ Z9 V6 H. |7 G) \3 G, K2 ^

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

: K+ X. K+ R/ l [) h

Edit1->Text=IntToStr(y);

/ S& T, x7 J) i0 x

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

e; V7 T$ s4 c8 A9 G

int x=6,y=6;

( Z$ n3 i5 R1 m. n! u

IMyCOMDisp app;

5 E# Z) C0 x8 F+ D

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

) ?) A- S8 ~+ T5 a( K, o

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

9 M* i; h* m9 T, c2 H* ~

Edit1->Text=IntToStr(y);

" ^+ Q5 ?2 S( H/ k/ B# R6 p

4.小结

( w1 p: {! p& {+ u* N; K( q. A7 W0 s

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

8 G* c* J; i4 t# r2 W4 x

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

' W; N* P8 n- O+ S5 F! _+ r2 ~; s9 H

" K0 F' N W, m3 j

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-6-16 02:29 , Processed in 0.503706 second(s), 51 queries .

回顶部