QQ登录

只需要一步,快速开始

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

COM应用软件开发技术

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

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

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

蔡倩

% Z; K: R# e( `, X1 L+ @" f

主题词: COM ActiveX C++ Builder

( D8 \1 l1 C$ c' D

1.COM技术概述

, i. \% s2 v- N4 R

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

L# \7 G( ^6 a$ L

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

/ @& F' N% O: ~5 K3 m

◆用于创建ActiveX控件。

' ~7 Q0 Q" r9 q

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

: n2 A% [4 G8 X4 F3 s M

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

+ F2 f3 a4 Q6 w4 ~1 h# h+ u

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

: V' n _' r" l+ X7 v5 q+ g& ?

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

' U8 J+ L9 Q( D: B

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

2 _9 `& g6 p6 t, `9 w

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

9 f. s8 w8 P2 w7 B' @: R

2.建立COM服务程序

, V/ x$ H. f$ K1 }$ F" J; ?

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

+ {0 C) a5 A8 n

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

+ Y. t5 m y9 A' n% w4 ]# w1 B

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

3 I% X8 B7 c6 t* X

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

- ?3 T+ f) U1 Q! C! {/ O

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

9 _- H6 j( E$ L0 o- o

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

' e4 }" k1 O8 J

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

. }; A( q0 Y+ D4 i6 b

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

3 L0 N! u" ]% r4 [, X

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

# ~# H+ d) k, J" i4 b

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

* e# F. P$ d5 |5 V) S

2.2建立COM接口对象

0 h7 ^# V# B9 p1 L

, Z; S# A6 m( L2 k" ?6 d

: w/ u; {) b+ K% I! y5 y; U4 i

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

( Q% D1 P0 s& c1 S, S4 z

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

+ f. g* a# H0 |9 _' q

4 ~8 P9 w: {0 R; O8 I

$ T9 U6 Y5 ?6 V2 ^" s3 {7 _- ?

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

" }5 G7 M5 m% i% i9 H& Y( Z& z

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

4 A) J* h5 |* e. C

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

/ B3 `2 P& {* g' [9 X$ h* p

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

: n3 o$ i" a1 m+ i0 U" E/ F/ O

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

! g M4 @5 F: R% H3 m

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

, B; v2 x) z) ?! w/ g

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

; `' o9 B( T6 |4 {1 F

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

4 V& Q; R3 i( }9 e) |

[id(0x00000001)]

2 p( J& K, f( y+ t9 q/ u- j6 w0 u# Q

6 g2 Y- q0 r/ |9 j

9 X- }$ i: d$ W9 o1 u: L

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

+ p% u I7 f: ^8 k

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

3 U. S( \: K( z1 j5 I: [5 r% i; j

PCOMServer.TLB: 类型库文件;

: X' \8 @) _/ r' d- d

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

" f& y" X4 D4 s9 y D

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

3 S3 h$ o: v, ]) V

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

8 E0 T6 v7 C/ n5 _

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

" g# b* D: T. E" [0 G2 S2 S

2.4 实现COM接口中的方法

: ^0 c$ E% @1 s( j

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

# c$ r6 l2 O' R- v. f) ~$ `, Q

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

9 D7 q$ O: f) m. D! u, w) N' z) w* A: x

{

8 H @9 Y- L- X2 {5 L

*ret=x+y;

1 m( a5 J2 A: H, Z3 a

return S_OK;

: e( c0 p( i, R8 ^8 P

}

: Y4 z/ J& T8 w/ b0 a

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

7 H4 v* j4 V1 ~& `3 E

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

; x! F1 y+ y. K+ j

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

1 w( i) [ z% ]$ m+ y

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

. c6 D3 V C9 n" E; w( ^) z

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

- \5 Y& y( x. m. ]3 @# [ P

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

# K3 f0 }4 y L

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

+ {' K* w+ ^7 x! c% L& G# y

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

4 [. Q$ w) d i" c1 v* e; m, e( x/ O

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

$ r. h) R6 J- X! F

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

- u3 ?. m$ e6 E0 T; l

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

% x9 F8 ?% B! p0 y I

3.建立COM客户程序

: M( b4 n3 c2 s) ~4 s

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

0 b& J/ u1 P2 x* F0 |1 I$ ^

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

1 Y3 m5 q4 q1 _4 X

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

; I1 Y: k3 \& ?

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

# n+ d, |& h5 E9 V0 w1 E4 O9 A4 q" ?

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

0 W& u3 a9 S) T

// Helpfile:

5 Y! g$ N- e/ p1 @3 s

// DepndLst:

, T0 V4 ]- G! `2 a, \1 R

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

) h, k$ `: [0 w, D4 V. q! J

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

5 \1 g" y! p0 @, C6 G# d

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

" @# c/ @5 X2 v6 i+ h# c

#ifndef __PCOMServer_TLB_h__

$ M- z3 j- d9 e4 T( K- j* d

#define __PCOMServer_TLB_h__

% ^7 N# W' d9 M- a, p) Z2 e' S4 b

#pragma option push -b -w-inl

4 H# q ]( \0 o) ?+ m

#include <vcl/utilcls.h>

" N- C4 t7 e7 I* }6 J) V

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

/ q& R% G4 r+ F# z3 I1 o( M& Y2 c

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

. A' Y. T% ~7 n ^3 O

#endif

5 T* Y' }) Z7 I# W& w4 ]

: z: H) N1 G( h0 c

#include <olectl.h>

- o3 S) u( M7 Q7 n3 C

#include <ocidl.h>

% P% `' w% Y' o0 K1 t ]

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

/ Q3 a% |4 E& [& d

#if !defined(__TLB_NO_EVENT_WRAPPERS)

4 U6 z* C) k' i5 N

#include <atl/atlmod.h>

5 \$ {" j+ }8 c3 Y

#endif

/ P( I# ~0 v% e% r; x

#endif

/ {' ~8 ?; Z: p( O, G- L9 L

; q( J; e4 O4 k. y+ C- Q

namespace Stdvcl {class IStrings; class IStringsDisp;}

9 X* M J6 L; _) L9 R

using namespace Stdvcl;

6 x( m( i1 J) ~1 Q

4 u& e# U: X& W

namespace Pcomserver_tlb

1 s1 V4 n- [# @' W" h

{

6 r4 p) _8 y% u

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

6 s7 |- K$ R7 ~5 J% [' u, y

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

# b" V) f0 M: O) ]

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

0 K9 G1 f# i5 c N

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

( b8 U; H) q0 u6 y

typedef IMyCOM MyCOM;

; P+ L3 C! a2 Y

. L" C$ D* J) o0 H. X

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

" S; y% k! z8 J) ~* V+ z7 O9 ]

interface IMyCOM : public IDispatch

: E% c/ ~4 x- @1 @" w. Y6 g% [* `

{

1 W0 i% B% ]; `4 G6 r/ R

public:

% O4 s6 i3 e. Y' V$ N- e' m ]1 y6 |4 z

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

L1 r5 f+ j4 j0 c

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

! n! u6 [5 T+ w) ^9 m5 K

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

; _% r ?# z9 H: f. e9 t# p

{

! f1 \ a6 t& `

int ret;

* N1 ]% n" I8 y& h% U6 y

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

# H4 T3 e' ?" b; @" \* D* |

return ret;

$ t1 B9 z5 O! j$ U

}

! c0 {7 I0 B j* F6 N/ M

#endif // __TLB_NO_INTERFACE_WRAPPERS

; y- B5 K1 |# r" `% e. E

};

8 l' Y" }. V5 c$ e1 g* e9 L5 w

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

. o% v, _$ z% c m/ Z2 Z

template <class T /* IMyCOM */ >

. j% l u# F# U9 X

1 \' [6 S: U0 A4 J) I: i

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

1 G H, K! q' H+ J$ s! a4 s3 i1 e

{

+ u5 f e3 Y, W

public:

+ e n; J0 Y2 J

TCOMIMyCOMT() {}

' G. }/ E& Z9 t4 @

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

. I2 H2 N; \% Y5 A6 V$ Q

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

4 r* k; Y# w4 S* c) [+ [

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

/ x0 o$ Y s; W9 o

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

: m: Q" ^- H! J) F( n" y" c+ k

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

6 k: V- o+ b( `+ T# }, O

};

8 P5 _) n6 q. [: I: ^5 U) A; S

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

! G" t# E6 o2 g- Y+ [$ t& f6 O* [8 [( I

; \% v3 X8 _; [$ i

template<class T>

- Q8 S. l! @9 ?3 U

class IMyCOMDispT : public TAutoDriver<IMyCOM>

% d$ Q) ?. J' y) n

{

6 y Q K1 x& @; {( J

public:

0 S3 P# v# Q* Z4 v7 I# ^. G

IMyCOMDispT(){}

1 X. _& {9 K0 Q+ A

IMyCOMDispT(IMyCOM *pintf)

8 p7 Z" [$ |" Z( y# o

{

- R7 z: K. b- l- g( ^8 t

TAutoDriver<IMyCOM>::Bind(pintf);

1 G/ y: i) ]/ s0 ^

}

/ b3 ]' }' ?6 R

IMyCOMDispT& operator=(IMyCOM *pintf)

! J) n1 Q' Z5 q' @) v

{

/ R8 F% D3 l% S' I; c4 U4 Z' U/ n

TAutoDriver<IMyCOM>::Bind(pintf);

% [( h. q4 e- a6 d4 J

return *this;

- e$ J* y) o9 \; g& O9 b

}

: F2 T+ {, ^" w$ F0 D

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

! U3 W) I6 G8 {4 Q' @: z5 B5 F

{

! G# b, L5 V& L8 \3 C

return OLECHECK(Bind(CLSID_MyCOM));

2 Y2 ~ |& k* }( N! ?) p

}

; u0 \8 p' I$ B. k5 g. u

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

6 ^& ^& b3 D+ f4 h' G6 Y* ]; [

{

0 _1 H/ B x# T6 i

return BindToActive(CLSID_MyCOM);

( c- p W, X R' a

}

( B5 F6 b7 x& Z, U3 O

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

9 B2 ^+ B n5 s

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

- r% x( n/ n+ M& j/ K8 e

};

3 h- |" w& }4 u

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

# @4 Q& N; C# a1 K6 N+ P; S

1 n) P; @1 o7 r) ~; v' H7 T+ V

template <class T> HRESULT __fastcall

; w1 ?! f; ^) _3 D& D

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

" ?* f' F( d& g8 s# _9 X- G- |

{

8 u2 @; W3 _: a* ]: ~$ S

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

. z' u% M) z8 I' S( s

}

. U( S* t; j3 h' E0 W

& x: ^, A# f1 r5 w5 r

template <class T> int __fastcall

5 @9 E C) r4 _" ~/ q

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

0 w2 W' Q1 ]9 z1 v& P- V+ D& z7 G$ M

{

: t; g' P: F0 V* ]

int ret;

& S8 u/ ]5 c/ p9 H4 X

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

) w' q4 b6 N: K$ ?( [

return ret;

" S& y. W c; O# Z& k( w

}

% H% O9 _; I/ C

/ Q! z5 L. H2 A

template <class T> HRESULT __fastcall

1 U' v: m O, p

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

/ o1 j+ g) ~8 P7 b6 Y( j

{

4 ?# `$ j$ P3 |9 U% N! L8 z

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

1 x! X& w/ R; v/ `

TAutoArgs<2> _args;

8 }& r& U5 J0 B, i d

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

/ J# X H/ K8 a" C& j2 h8 V

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

% e: w+ D5 g; m" v- F3 X

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

% d4 ^% o+ A" d+ |* a& |; P

}

: {+ k) l1 }; b# i, m8 R' G

( R$ ~# L" S3 w) |

template <class T> int __fastcall

" q [, x: Q; q* e4 e4 A3 r4 J

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

. ?* O" b2 @9 K; Z9 ~- y

{

1 q" n' S" i2 \% O! P8 a$ f

int ret;

& \, [) @6 J5 Z4 f% _2 Z& D

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

$ z8 G2 \/ Q# Y+ S% C

return ret;

9 ~: R9 ~( \: K9 P/ s: S* D9 [/ L

}

3 R6 I4 Y; }3 O( c$ q: {

. A; n# `( k8 M0 r7 ^1 S% [

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

0 }; e, R8 w' U3 O2 a

#endif // __TLB_NO_INTERFACE_WRAPPERS

' O' K2 B; }' R- h t! I

}; // namespace Pcomserver_tlb

* S; [% c {7 @* Z1 P6 T6 T

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

2 [) E k7 H! k) X3 ]

using namespace Pcomserver_tlb;

1 }% D+ K# t* u, n j

#endif

; U; l3 ]2 x$ Y

#pragma option pop

9 F9 E% J: F- X j

#endif // __PCOMServer_TLB_h__

7 {# j$ l- Z3 `0 t

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

+ q) N4 q# e# h7 Y+ ^' q+ ~

interface IMyCOM : public IDispatch

8 @& @3 [) G8 r0 [' |

class TCOMIMyCOMT : public TComInterface<IMyCOM>

" I6 { W8 M# P9 ]+ d b0 m Q

class IMyCOMDispT : public TAutoDriver<IMyCOM>

# |! E+ w9 q7 |3 E! H( g: `

class CoMyCOM: public CoClassCreator

+ N! S' [# o2 [9 S( g! d

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

" p! Q, K! U$ ?' o* d+ _$ m3 n% @

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

& f$ o5 _/ Z9 Q3 t" z

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

2 z* b# S. d9 z" B

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

+ H3 h8 r5 N4 n4 \' W, e

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

! `& w% w* N3 z! G2 F7 Z* e- V

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

% Z/ _ D Q v1 m# Y1 d1 C e

int x=6,y=6;

+ S3 |: _; m$ C5 [* c: U2 v

TCOMIMyCOM O;

- K7 @3 [6 ?) t- t% I( p

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

- h9 a3 W% g, y, @. H: c

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

' Y) g, p1 q c( `/ q! T

Edit1->Text=IntToStr(y);

: k6 T g0 f! V& s

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

/ f2 {8 @. W2 c, u5 M$ d# K

int x=6,y=6;

, t0 J l9 A# {7 x+ A

IMyCOMDisp app;

5 M7 {+ Y- d. I% G. m, E/ B2 I

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

5 G' Z4 v3 |' l6 G( G

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

/ P1 _: S, h! q8 l9 R

Edit1->Text=IntToStr(y);

- d0 N' t0 @. W7 _7 i# [9 O" N# p

4.小结

) ^, Z' U9 Q1 d% v6 b

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

( q2 }0 ]1 w/ b

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

/ R1 t2 x3 e4 D, J' E

) s$ n' U2 E9 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-18 15:48 , Processed in 0.433524 second(s), 52 queries .

回顶部