QQ登录

只需要一步,快速开始

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

COM应用软件开发技术

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

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

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

蔡倩

3 k) Z) O8 b0 m/ b I) ?( Q: [

主题词: COM ActiveX C++ Builder

" a" ^7 B, y) h0 J. W

1.COM技术概述

1 Y: R9 s. p- y- s% a

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

0 w8 s& h' u0 I; d! `

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

- H* h4 P- g; k: n/ G

◆用于创建ActiveX控件。

i( N& M5 b" S% Z

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

0 B8 w: q( z& |( [

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

: M" _: q+ X1 \, c: P

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

- C0 G3 `# D, \- r8 ^6 O

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

( H5 J v% [! j, f" ?

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

6 F4 z& x( o7 x5 u9 l

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

. x& N7 m( }9 t$ `: z

2.建立COM服务程序

; w; \/ c. a% G

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

. P9 X! |5 V c9 a

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

4 P# @, d* }) ^$ g# P

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

; p$ L4 W" w( b: N* h

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

2 D+ Y0 f, h* L1 r

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

) T- G5 r( _. B% D" R& T

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

% i- A2 y m9 K* |0 C

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

& y; N9 y8 c% ~' m9 v

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

+ V+ Z* x" a& K7 S" M- Z( l

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

& p3 x6 e( l7 W5 c* \) |0 _4 O1 l

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

# s5 M T) o$ |/ W! k) j! {% p

2.2建立COM接口对象

# `4 h, z! C( R- p( i8 h

* [ k& x0 S4 \+ U/ i% q# j! A- R

, o; V, b; @1 b) K0 T% x% }7 ]

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

% p- ^# [4 A& M* M7 p4 v+ i: F

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

5 s# l+ S; {# e# L9 |

0 B: _, s8 i) A6 Q

$ S) P- X m7 ^: i% l( v) I

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

: [0 e7 d, ?+ D( T$ N

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

& u: U0 b) k2 P& S3 N6 M1 h. v

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

2 T, z; ?7 ? z+ _. u

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

' M' z2 d& e8 b) s

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

( N) b: b' l3 I: K

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

; N4 _$ ^5 l' @7 ~; p7 V7 S s

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

) I( u3 F" K/ Y, w9 Y) C3 u, O! \3 w0 z

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

1 I1 s4 o7 r- J" ~; V, Z3 F

[id(0x00000001)]

* }* _ z9 ^! Y# K v( L

! G# S* ]8 X! s( Y1 k1 l0 p* P; x

# D; e" h o, u2 b3 T/ u* x1 l

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

3 S7 c1 U5 b2 f5 ]" V4 U/ {

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

$ i7 @8 X# T! q! P9 I+ p6 |* Q4 {

PCOMServer.TLB: 类型库文件;

: W6 Z# Y5 O- @& b2 ?

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

& r* U: H8 J! y1 P0 g

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

- n0 n! R( f5 A& F( t/ _

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

5 S+ i3 b4 `& R U

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

6 S8 |" q+ u. }1 K- Y8 Q" V, S

2.4 实现COM接口中的方法

9 `; U8 r* \& A' [* e

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

2 h- G: r" n+ ?

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

. b8 G1 P# r7 W& F

{

2 a0 |& e# L$ m

*ret=x+y;

. j5 B' D7 U: i

return S_OK;

; P3 Q% q/ }9 e+ b& o

}

4 A' C f3 ~3 e

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

! r3 `' x! B: J: i% Q' V4 m0 P4 g

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

. I3 h) I0 B7 t; w- k# X, B2 l

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

9 N" L4 L. q v

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

8 ^5 u; M- Y" w8 p

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

6 _* J3 ?$ m g2 V

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

( Q' ]1 S( `4 K; R9 K* }

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

# l& k" M' G$ E

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

: q# z6 ?2 O4 ]4 H; a

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

! E# c8 m P% x

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

5 F, C4 c- f6 T4 ]# R9 y, k) v8 I4 ^

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

8 r6 @* T, z$ @; H2 t8 ?. u: r- F: {

3.建立COM客户程序

9 ^- {' I- i. T% S

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

. m( }) b7 ~# U/ X$ y) ~

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

) |( l. N: [. G |) K5 C

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

7 F2 x: \: K6 a6 t

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

0 t( H$ F1 y, c; \# z" b; Y5 ~

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

: i0 g L% B- m3 L9 H8 d5 @: U$ p

// Helpfile:

/ X( z$ ?$ V* _- @+ R1 O- ]9 c

// DepndLst:

3 w0 r0 [, g" i, I

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

9 H- t- p) e/ N. I8 H9 V# l

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

5 I( M/ Y, u6 ^& l* g& \: i

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

. b9 X% \6 \; P

#ifndef __PCOMServer_TLB_h__

/ f/ \' \1 ?& C

#define __PCOMServer_TLB_h__

0 B6 L3 k8 l$ h5 x: c2 e9 n

#pragma option push -b -w-inl

' A# v) N r7 r; W

#include <vcl/utilcls.h>

' D, ?# P# n9 B) ^1 {' J" w

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

8 _% S3 V- P- A

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

) N; ]# D; a* o: h

#endif

1 l( o- E! A" W/ @# j

, C1 H7 X; k$ b$ [8 o& W1 ^

#include <olectl.h>

9 H* F( B8 ]0 _

#include <ocidl.h>

' R9 B! e& e. t# L+ E- U3 J1 @

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

: m3 B- D; J r2 K6 Q& _

#if !defined(__TLB_NO_EVENT_WRAPPERS)

7 v6 ~+ R0 p8 _( K# h* ?2 @

#include <atl/atlmod.h>

# g6 Z/ u1 J6 P

#endif

: s3 p: d' L, Z' H+ P% V' D. F

#endif

, _9 `3 t3 t: _! J

. F3 }3 {( S+ X# }* C

namespace Stdvcl {class IStrings; class IStringsDisp;}

; [" |" s" h8 o( R. B8 c; F, h8 O

using namespace Stdvcl;

# W4 I" r. ^7 B/ z A- D

; d8 \, \" s1 E0 ^ b5 g& z

namespace Pcomserver_tlb

7 v* d( [$ D5 R# p/ z/ ^% N

{

, d0 z" N2 g& O( G+ G9 v

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

# v. m( W& S- a% G. `

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

- O) n- F$ \( [

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

, V ]$ l( F- I# X! l# ]

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

& U8 r5 \* ~3 P+ O0 A; w

typedef IMyCOM MyCOM;

' g! z' J" Q7 V" V I4 s/ o

( z2 {% `& Y& l) a& k, B# p+ A( a* r

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

4 L% o/ d' o+ Z* T7 [& L! q# A0 S

interface IMyCOM : public IDispatch

0 C, F+ [% L$ ?

{

3 W( r' Q8 {& Y& v8 z" M

public:

' [( Q+ X$ E( F6 J/ ]4 N

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

8 d, Q8 i' d( Y2 C3 T. l! h" _

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

* z+ k1 ]+ S+ n; K9 O# r5 o

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

+ E* N. }3 D- D* f- P$ b* s

{

1 ^4 B0 p8 ^ |+ U

int ret;

& |$ J3 C+ v' H0 D

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

' j1 u3 [! {- l( \# u I

return ret;

7 i& t% J& `- u

}

! t( c/ n& y5 W3 u8 a: v

#endif // __TLB_NO_INTERFACE_WRAPPERS

0 b: y- B! T( X+ s" `

};

, n X {) g* A9 {1 Y) L2 b* ^

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

' g% m& w" R+ j: ~$ R$ e4 V

template <class T /* IMyCOM */ >

, |& q4 X6 z) x: i- J

9 p: U B/ E3 Y: d6 S) X, d

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

9 _" K H* ]+ B( h. c) I2 n

{

6 w# p2 T% Y6 j* q

public:

$ A7 Z+ e5 }2 X3 U7 Y# a

TCOMIMyCOMT() {}

' P8 o) [1 c v

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

8 P2 r1 w p/ k! p$ ~, V, {

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

$ z) n$ C! \* D# s7 `4 X

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

% L4 ]4 ]2 Q F0 A: f# R* W% y

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

7 W) l+ _; P* W6 ]5 L; q6 P+ k3 x8 o

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

1 ^; T. `3 o3 N. k) R7 D

};

) P z( x+ x* V0 Z. f5 `

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

5 V d+ R( b( x# U9 i- V

2 H! I/ ?% Q; u( Z% u. c

template<class T>

; Y/ b" I h, F* Z* M1 O9 x

class IMyCOMDispT : public TAutoDriver<IMyCOM>

5 y. e" U& r( W5 I" W

{

( ?& F, g2 ^: R* E- W( x, P# x; s& b

public:

% z( \" p. u5 B# g+ y' n

IMyCOMDispT(){}

: I- m: n" V/ O' w

IMyCOMDispT(IMyCOM *pintf)

8 s! U, ]/ I! z! ?

{

r; A; n! b9 z0 }

TAutoDriver<IMyCOM>::Bind(pintf);

# I5 U E2 e, S( R z

}

# Q% i; a3 B% F6 ?/ }6 j

IMyCOMDispT& operator=(IMyCOM *pintf)

& e1 W" G3 r* x. H. D

{

; U e, B1 {% ^" N1 U8 ~% n1 c

TAutoDriver<IMyCOM>::Bind(pintf);

* ]6 B+ {6 f( t, P7 F# F

return *this;

. Z, F& x4 _7 l4 m3 N& ^

}

" q& k+ a4 W% W* n6 |

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

& a1 C- P$ S9 x- }

{

2 D+ }3 Q* i0 i" o4 S* w9 `

return OLECHECK(Bind(CLSID_MyCOM));

! t" q6 H" R; [

}

& [3 X6 p' _9 y) r$ R D, U, i

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

+ B/ j+ o! z. Z

{

* r9 k3 |/ c( z- T" f/ Q

return BindToActive(CLSID_MyCOM);

+ w1 l+ q! P' t

}

; |7 u/ w4 Z+ d' `+ o7 `( e( B/ n

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

. x2 _, }% I( q

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

* o6 x+ P+ W! q) Q2 q

};

: T, D6 P% U- g; J' R) R9 u

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

; j% ^2 H8 |4 D0 B) Z2 G+ p

4 L" ^! U {& c, ^4 {, f) A

template <class T> HRESULT __fastcall

& D" C5 `+ E& W7 c( @ r: V

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

+ f' l" e, h0 a. m+ L+ N w( i

{

* v# h. E- X+ h; X# Y u; h# z

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

X g$ X. s9 S7 z# X

}

, h# J9 U& Y; c4 H" e

* }3 M8 ]8 T" H8 k' U5 z

template <class T> int __fastcall

$ ?! N! n. x6 ~

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

4 W; G6 k4 u2 a5 G6 X

{

' n! D" y; s7 @9 K. `3 `' H; s9 \

int ret;

2 P& Z, N& _! m( e0 \( s# G

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

' z" |% W' Y; {! n5 y0 C {0 c6 h0 L

return ret;

/ F4 _/ R7 N) `* Y. p

}

8 R, B$ ~% b7 x* e" e( J

& [ l5 o7 q8 E# \. H2 \% k, C

template <class T> HRESULT __fastcall

4 f9 \/ l8 C7 F$ R

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

, n) R# F# O. f: E' o1 h

{

~0 T. Q" m3 U2 j

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

" |! j$ n' ?/ ]2 V9 i

TAutoArgs<2> _args;

" Z9 v6 {" b0 ^; i; y! B

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

' R% ` o! w. B- k& N( B3 j

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

5 j* x- R5 M. U1 `* z

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

. T$ |% ]3 J$ i% ]4 u

}

. } B0 G2 b( B' r4 k3 {

2 l2 t+ S! q7 }- L) p: k

template <class T> int __fastcall

$ ]: n# ?- O6 U6 O9 m

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

_% W7 s9 X! g. j/ }. |

{

! d0 u2 w; S7 L8 C( a& G

int ret;

* s) l: S& P& W0 k$ c) H. o

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

9 s" X5 y8 Q5 [. W w

return ret;

# v& m. q, N. A2 H+ Y/ B5 a

}

0 k. s2 y/ L/ `$ n6 q& k

, r, c$ y0 k1 G( B; H; R: w; ~ l

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

" V( z! `" L# z) R

#endif // __TLB_NO_INTERFACE_WRAPPERS

L" f4 c5 ?$ v2 V' A$ v- j' y/ k+ i

}; // namespace Pcomserver_tlb

. t/ z* l; B. | ~$ [4 ~) ?

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

% g* ~+ K; x& t7 J* S" g+ u" f. p

using namespace Pcomserver_tlb;

, {. u2 L) W& @& m3 i

#endif

, }- P8 h) g, ~; a6 k

#pragma option pop

) i* G( E7 @: m/ K8 ^( s/ P

#endif // __PCOMServer_TLB_h__

' H' D, F3 b8 {3 C) H8 G3 b3 U7 F

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

0 Z5 U2 B% Z& F) n! ]9 x; t7 Y

interface IMyCOM : public IDispatch

, `- ]2 O, Z8 a+ B( C

class TCOMIMyCOMT : public TComInterface<IMyCOM>

5 p* Y8 X! o% r7 i% R

class IMyCOMDispT : public TAutoDriver<IMyCOM>

: i; @& w- H# y/ M& m

class CoMyCOM: public CoClassCreator

: ]5 X5 d5 H* V1 ^! C

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

. F- J4 t) V9 w7 O- H

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

, h4 k" s2 _; |0 S

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

0 q3 o, p, m2 e% ~, G% L+ X

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

4 h: ^$ C6 I( A& r0 U7 O

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

3 A9 ~5 c4 H; g7 _# o/ N: O

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

W& f/ F4 Y2 b

int x=6,y=6;

* t- ]* n! s6 O- q7 [5 A5 h

TCOMIMyCOM O;

- A5 e4 N1 Q: D2 h* Y" v$ E

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

" e( u5 T& w8 x1 \8 ?* @

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

3 a7 j' M" \2 S& f

Edit1->Text=IntToStr(y);

, r q: T! _- y; Q p6 h; K2 F( F# N

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

$ D7 H0 U1 B j; R* ~

int x=6,y=6;

& f5 V- X$ M" y% Q! b2 @

IMyCOMDisp app;

. k* \+ Y( y+ i# Y5 Y- Q1 A# {* n

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

- x& d: [/ T3 J1 _0 Q1 y1 d! s

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

" p( R2 J) z" b+ e) T) c) j

Edit1->Text=IntToStr(y);

2 a% c% o: G f" M' q

4.小结

! g- h; J/ E2 A9 H- a; w" x8 t2 U

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

, k$ A! t# A2 `( t6 z; m3 ^

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

U$ ]7 B7 j2 V: [' d

4 d4 U+ X3 t; q3 K! `9 O

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 18:18 , Processed in 0.459287 second(s), 52 queries .

回顶部