QQ登录

只需要一步,快速开始

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

COM应用软件开发技术

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

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

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

蔡倩

6 [! v* ?# @& k- S0 G: L+ y

主题词: COM ActiveX C++ Builder

( C: p# n j$ }0 e) W

1.COM技术概述

$ B" g/ l6 U. P) H$ k J

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

6 W1 K- @; d% C5 M1 R* z6 l

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

+ I0 N$ r7 }$ t w4 ~! h8 ]

◆用于创建ActiveX控件。

6 F: m- J) N9 [

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

6 V- p: d% M' A: X2 M

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

* ^, c/ R4 j7 l( B K

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

$ ]/ \( P5 U$ I0 w

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

5 p0 @% m* L* g- W) ^

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

$ G+ F$ P3 }+ @+ w2 I2 m

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

/ t2 t" p+ J! {; @/ r1 ^6 ~

2.建立COM服务程序

: h8 ^4 _% D% S% _

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

, V9 H9 H) x; k& G( P7 b7 \# I

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

0 \' Y$ f- `" X9 E0 [( F( ]7 D: \

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

/ d& u3 v: B; ~, z4 f7 m6 ?

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

. p2 l& E5 I+ q4 X

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

8 \3 J; X% v5 W& s& _

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

% K( j S% x4 @2 m n$ Z! D: g

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

' I+ e9 k: }6 m9 p2 l( c) V9 N8 I

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

; i; S: r; s- z. r

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

/ O& c9 a" ~& ]5 z* m) M

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

1 v' v, f" @. t2 z2 w% g) |

2.2建立COM接口对象

8 t# E4 z: k! ~5 D; w9 T: A1 x

- t/ d* f1 u; f

, M% f3 M* L" k7 @8 C8 J

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

0 w% N% ~% A0 p( u

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

5 g1 s2 U* [# T8 G6 A' H: T/ c

1 V; L1 ^6 P7 Y% W3 U

& M9 [3 `. a$ C8 w6 w- ^3 O

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

( T' S9 M9 T4 U& B% B

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

2 D" X4 c7 b( E& `" u

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

; ]! F2 }$ D3 ^

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

" C T7 Q2 H0 @! g& h

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

3 L9 L! ]$ ?+ P+ Y

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

) j/ E+ T& H# |- H- x

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

" K% |. e, @8 @$ f7 j

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

) W c1 J" w* X/ V0 a$ G( O2 d( V

[id(0x00000001)]

, k3 j8 S$ G" ?8 O8 Q& n

' E" t. [3 {* @6 N3 W

# ]( a6 V. _( F

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

/ h& N& g7 b8 l+ e$ K- t7 ` o

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

0 \3 ]! V3 E5 e0 B% M

PCOMServer.TLB: 类型库文件;

- M g* C: C( _% z

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

; [# x- M+ I- _5 [ v i

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

( f' _* w, @. C$ h

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

" y0 n. i. V1 F) A% c

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

. O+ [, ]. S+ j w% ]! R6 I

2.4 实现COM接口中的方法

6 x! o) y* p j' t7 F

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

1 G1 Q2 \! d+ p* @: R' |: P

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

+ T) ^9 v% A% o. j2 W

{

9 d( \& Q- p" P, z% \0 U

*ret=x+y;

+ O: A1 J- ?! S! \2 J9 ]+ Z" j

return S_OK;

0 J1 o( f( J0 T, o/ t, w

}

- K( L2 }6 w- O! }

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

$ E2 O) z* i# B1 e- U2 v' n

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

7 l0 c/ T3 E9 o, L

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

& e6 z: E1 u6 f) O8 o

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

" X* i; N* h) R

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

9 s6 G. r/ a0 _8 C: _, P

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

! a0 E: f$ J1 k9 j q) x1 N

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

y5 }6 R4 Q0 \9 e Z

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

$ X# z/ H8 e! v4 V& v7 a) S- v5 M

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

/ `2 B; S1 R3 g; p2 B% e6 Q

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

* M5 Q1 S9 T' Z( ?5 f

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

% a% ^- k9 B. ]+ b- X: A

3.建立COM客户程序

5 z( S& A; W! A+ [. k1 w

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

+ A0 C* f n& p7 R6 p [- b

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

( ]8 k" j! ~; K

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

9 K3 M, {7 q1 p p1 {* ?; i* U

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

: @7 g2 Z: V, |, r8 \' x# I

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

! Y6 t4 G" ]+ C6 K1 r+ Z

// Helpfile:

7 e) R7 t4 _6 Y6 Q/ j

// DepndLst:

0 o) h( _3 _& B" [7 }; R6 t" ^% P

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

4 D. X) p+ p3 X0 T" n1 G

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

; \! g) T5 ]1 B0 E

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

: Y" Q+ M- o& o- B6 j, ] i

#ifndef __PCOMServer_TLB_h__

F2 |' k0 X( I, z `6 }

#define __PCOMServer_TLB_h__

+ ]. S+ K, u4 c/ ^ X( R5 o: Y9 X

#pragma option push -b -w-inl

4 A# n/ Q! {- y0 x# [0 a7 K# i3 P

#include <vcl/utilcls.h>

, I3 K" G; D& X3 o( N

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

+ ~+ ~5 F( n a; X' s( N8 n

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

; M9 p, y7 {' q; Z2 ~

#endif

4 L/ {0 t) u# p9 j. R% {

5 z" _( R6 u2 o1 y7 g4 U1 S

#include <olectl.h>

; w! s D2 W5 |& S' s3 Q" ?

#include <ocidl.h>

+ T4 W8 B, x% H0 H6 {5 `

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

: B# Z5 L) x4 v" k8 N2 N

#if !defined(__TLB_NO_EVENT_WRAPPERS)

8 }0 }, q/ F3 N/ A1 z. C

#include <atl/atlmod.h>

! D8 b& ?' p# I6 K+ g

#endif

y7 z2 a% {" q" G4 P" V

#endif

$ G" L: p; {1 Z

. O6 s/ h3 y; l9 H& {9 ]

namespace Stdvcl {class IStrings; class IStringsDisp;}

! c; U5 m" f0 o7 f8 W* j& [

using namespace Stdvcl;

) L) z/ I& ?* `6 c

# p5 x8 P" {! f, ^! G

namespace Pcomserver_tlb

2 U0 ^1 J+ j% R2 b9 P

{

) ?4 ~: s# Y+ g% L

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

# i8 o" V% K( J

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

) ^# k! q- b' _; k

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

$ E: x! q( o2 I/ S/ ^- V

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

, N" d1 ?9 H- b% T

typedef IMyCOM MyCOM;

8 L9 ~7 O2 y& Y ^ J" A% @7 _

4 ~9 K. v+ \, z9 M. |

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

2 I3 e$ _1 ], o4 |3 C- h

interface IMyCOM : public IDispatch

6 z; N! L4 w7 l' ?* f2 W. S1 o

{

5 f' Q" S3 o. ^! x! c

public:

; y0 X5 L2 l) a1 l

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

0 G- z. i$ o; c

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

% l4 |; q0 g6 c! w% E1 l

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

A# g& b" N5 c3 T7 }4 K8 ~

{

0 G9 J8 \1 E z

int ret;

: A6 l9 M3 [+ q1 K n

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

, L" b! e; s; a

return ret;

# [" ?4 W ^# Z% s- T

}

+ A6 t! N9 {: a T

#endif // __TLB_NO_INTERFACE_WRAPPERS

8 d& r- P0 t9 @: p7 ?' ~, w

};

' z1 D$ Q+ L2 s- `4 C2 z% J

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

& u$ f0 t. B S: o

template <class T /* IMyCOM */ >

( F3 O. P$ `% j" H# B

9 w1 U2 P+ K! n5 I9 w$ d

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

. P9 e2 Z9 n2 @) u: e

{

z. B z }0 F- a" R9 G5 C1 ~

public:

6 d" J4 _- @$ }& q

TCOMIMyCOMT() {}

% z8 y2 H+ \: |) S* x" `

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

) O7 L, R9 R* \6 C

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

8 U: ^, G D! _5 }* b

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

7 |9 ?; N7 C P4 @6 O4 o/ b

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

- g& q: l: \ p

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

1 |/ ~$ N. E! Y N& v

};

% v+ Y3 R7 a% z9 c

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

7 p! z# L% e* N# E' F+ R

% g' A1 `' m$ @" v, Z7 |5 b5 I

template<class T>

% r# \# n( o$ e8 t1 U$ B) i

class IMyCOMDispT : public TAutoDriver<IMyCOM>

( Q1 t5 W/ z" G

{

( [6 d0 F# Z. y8 b# x- U' a8 W

public:

2 D0 [# F5 q+ |, Z1 H+ X

IMyCOMDispT(){}

- B# V! Z8 T! H o/ \

IMyCOMDispT(IMyCOM *pintf)

0 k- K$ @. ?, U" S) O" F

{

8 c$ p# i: F: [ S+ s

TAutoDriver<IMyCOM>::Bind(pintf);

4 e4 G* T: x O

}

4 d; b2 }1 v0 K6 g7 i* u7 S

IMyCOMDispT& operator=(IMyCOM *pintf)

- ~ Z9 i2 Z# u8 D

{

C# s& U# X- [ |( `: z1 \! [% E

TAutoDriver<IMyCOM>::Bind(pintf);

: N7 f* K7 i+ _# q ?4 U

return *this;

0 j" ]/ p+ s4 n" b4 T, b2 i/ M

}

7 E/ O0 z! _( B9 x6 l9 l( n

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

; Y2 A, m) E8 O) r

{

2 V \# l0 F$ G" s

return OLECHECK(Bind(CLSID_MyCOM));

- [4 K" [2 C- u6 q5 w6 e" k

}

2 N0 j6 ` @2 L7 h1 o2 j6 l

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

8 C2 `& G# a" N, s

{

' H$ H) k) M% z, \0 ^( i/ C: v

return BindToActive(CLSID_MyCOM);

O& ^3 o5 c0 j, h M

}

1 n/ W. G' {" Y. k! J

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

& k( y) J% C2 y9 M

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

1 H g0 _' o: o% y

};

" |% ~8 h, [9 i) |& |

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

1 x8 ]( U; f I& ]

4 M: x, h) b/ D' y) c) f' \. \

template <class T> HRESULT __fastcall

$ U" x2 a( a- x3 |

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

1 K8 J( R4 x2 l# ^0 v

{

( p, ?: b% ~( B( s

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

7 J5 ?5 r7 H J

}

/ K9 S& k. k/ Z

5 r) Y- ?; U* |

template <class T> int __fastcall

7 Z- L) a1 j; v9 y0 p9 H0 u7 p/ j* t. q- T

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

& s+ s. |: V; [* f

{

; d4 |4 h( h4 _5 K$ u+ ~

int ret;

& ~8 A: F' a8 O: f, D& |' T

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

. m# s, G) `6 o- o& k& ~0 O

return ret;

3 }) A& c# a4 ]3 O' c% _

}

* T4 D, V$ z; b

/ u K! v) S' g) h! Q5 P

template <class T> HRESULT __fastcall

0 |% \1 N0 r( y+ i* D7 h( R

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

* E0 `/ A# m3 I

{

* N* `# d1 A; q& C

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

. o& H1 ^3 e; l7 t% F

TAutoArgs<2> _args;

" z2 l9 ], B4 D, l% ?5 Q

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

$ }: A, Y/ T; R- p# J3 H

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

& [% w, m O+ m0 E: g

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

5 w) B5 L8 o/ t4 D, A! M

}

3 ^( f: t9 f* c

6 b3 v4 }) e! x0 a6 c

template <class T> int __fastcall

5 P1 I* z" w4 Z: Q

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

0 F0 d# k" [6 v' ?( r) ^

{

9 S- q. S0 b' j) X

int ret;

; x7 S2 ?+ t& w

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

1 n/ q) h. O7 T9 x

return ret;

( c& h9 Y2 D+ d. l

}

/ c" @' r, F8 X2 l& i* n2 U

. d' V4 M V- W) b. J

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

% J; ?3 V, U1 h! G! @* P/ f

#endif // __TLB_NO_INTERFACE_WRAPPERS

4 _0 K& O" P1 M. ^. ^! z

}; // namespace Pcomserver_tlb

# N& Z7 O7 [" {& b3 X

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

) S' b. A. P0 H8 z) q2 l! u8 |

using namespace Pcomserver_tlb;

- [5 q" N3 n1 B. b, K

#endif

0 B7 d+ m5 V4 d6 ]3 Q

#pragma option pop

3 u, V4 S. Q* H+ b5 X

#endif // __PCOMServer_TLB_h__

0 f5 S4 l1 P" T+ H0 Y4 _

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

+ k/ T2 b9 ~# f( y7 y* o2 Y

interface IMyCOM : public IDispatch

' a k0 k5 K5 E; } J9 A4 h" O

class TCOMIMyCOMT : public TComInterface<IMyCOM>

" T* |0 n' t2 y- T. S

class IMyCOMDispT : public TAutoDriver<IMyCOM>

0 D( _: I# K: L

class CoMyCOM: public CoClassCreator

/ `" G& B- r8 s# z: q* B( g* ~

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

& ~9 B7 u6 I9 j& X& l7 ^

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

4 k' _: t2 {" M! L

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

% w; L4 `+ B, r

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

8 T, ^/ W5 I5 E

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

8 Z( v0 r1 E8 P$ M2 {

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

9 i, L/ e' x) s+ { c

int x=6,y=6;

: w% _+ h' Y1 s2 k. C) K

TCOMIMyCOM O;

?8 B! U& y0 U8 S) `4 i& L

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

& P% R# y9 ]6 [( y+ D

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

6 o/ g1 B6 N' `8 y; \/ t

Edit1->Text=IntToStr(y);

3 K: [+ Y6 P3 Z0 T" l- w) l. ]

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

9 J+ a* |1 e9 d* R

int x=6,y=6;

; J9 `' R$ t8 `2 A

IMyCOMDisp app;

: {7 f- b0 X$ m- i

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

4 G j) {& p0 b ~; X

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

. Q5 \; y' ]% P7 b; J0 F

Edit1->Text=IntToStr(y);

7 b* |9 R% l- @1 ]; Z1 V8 m4 K

4.小结

* _' N8 W# @" p- J' P* x

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

' N( i$ t* ^; B! [

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

1 ^* P) J0 c* J+ _

8 L' H4 m" T# T! Z, s F3 q9 M9 z

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-14 09:24 , Processed in 0.439540 second(s), 52 queries .

回顶部