QQ登录

只需要一步,快速开始

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

COM应用软件开发技术

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

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

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

蔡倩

- B$ K) g8 U9 ~. W B$ s I' P

主题词: COM ActiveX C++ Builder

1 d; A4 x6 G9 b2 _6 ^, [

1.COM技术概述

% `# D2 c7 z. f9 I' d

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

9 z4 t& h( [! T" Z& L" U' N, {

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

% c! b# R2 L* f$ `6 A

◆用于创建ActiveX控件。

7 y" j$ ]) p* ~( X5 [

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

7 W* | D R$ e3 k

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

/ K6 U+ V" H, e% E, J

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

% Z- `" W' G" A5 p! x* x) Z0 Q* n

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

' P! L6 R% F, U8 j' G0 P$ [

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

! B2 ~' c' E* z2 C

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

' W# s+ |8 p3 M& E

2.建立COM服务程序

% p. q( m' n6 M( J1 V

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

1 }& l4 ]8 _0 P

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

0 q: K( P% R; G

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

" B# t# L/ @9 P9 F

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

- D# p2 b$ I; e- F6 j2 }

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

1 [0 ~7 Y9 ?9 X" E1 V

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

+ ?# ^4 r: s) r" L

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

+ s# }) X1 l3 ]* t" u3 X

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

t5 f% g3 b9 ~7 H+ W, E1 R! V# _+ m. m# G

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

, [* R9 [: N7 ~4 [

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

8 P, ^# B' |3 Z9 N d+ ]

2.2建立COM接口对象

7 i9 t# o8 ^8 g! k1 [1 \6 j

& }: |( @- ^' U+ V# w% s

/ b0 \ n( m' r& W7 G

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

: s/ @' ]1 r2 h* X- l5 |* n/ F

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

# _ N$ t5 n8 U. w. S

1 |& q, N& C& q! `

0 ^! |, w/ k H

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

; d' Z2 N9 y5 }4 v9 Y1 r% g$ V

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

* E8 `9 {( _1 m$ L6 f6 c. e

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

9 ^8 G+ v- e; x1 y8 _ k

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

6 P0 N/ ~' u2 s z1 P

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

: Y! D3 ]- o' v* c4 b* ]. P

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

3 I+ d. @) d# P. q2 _

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

4 ]9 }! s6 Y5 }

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

; X* B% `; ^! j1 b/ b/ v4 e

[id(0x00000001)]

: H8 S) ^# o+ h/ `; L4 v w! h

" F0 G- @+ ~# z9 _5 W

& z; B3 M- d; e/ V- U; y4 c$ Z

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

+ D) u& V4 r( L1 t3 h

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

+ I3 x1 g) i4 |5 D1 a$ g! I, n5 V

PCOMServer.TLB: 类型库文件;

1 Q' ~5 e0 s9 _% B' N. X* K$ a

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

, Q+ G! {4 I( z+ S' `2 | O( v2 V

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

0 I1 r7 T! d+ r

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

8 @ y) J2 A$ @ H

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

) Y E$ d# O& a' w+ X0 {

2.4 实现COM接口中的方法

# u X, j& n, R& k( U* b% [

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

4 ~) h8 |( a7 x

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

4 ?; @; i1 I0 k; V! D/ T) k9 @( v4 a% `# v

{

' U1 r1 X1 s* `3 u8 `5 {+ \( S

*ret=x+y;

/ |% C; p" O+ o) H: J( A/ p

return S_OK;

% f- z- Z8 s, H$ o

}

j) q, t' v0 ?

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

0 S( w6 h/ y7 w) f0 x5 c4 D: `7 O

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

' ?! X2 ]' A4 Z; s# d/ C3 O% S$ d3 r" J

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

2 q( U* N9 i! r. W5 k

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

( Y1 R' ~- J0 D' ~4 G1 Z" ?4 M* Z% d

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

5 h4 ]1 o+ a, t% a# \1 T

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

- z& B0 s! e) l2 m

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

. ]6 o8 o3 }& Z. i! _

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

9 o8 h$ \6 {$ \- N: n. B

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

. U8 P! N! K% _. @0 M! J, g e6 m

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

- N" T @9 `( B- | {6 n

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

" R& |) X. {! T8 @' B% E2 r2 H

3.建立COM客户程序

" v( x1 L8 E3 A+ p- @6 N: a

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

. ?7 U% w3 ]( V2 f3 t

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

6 j0 I8 P7 J, e* o

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

3 W5 t* _2 ]8 u1 t' V9 O) G3 V/ U: o4 z

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

4 Q) r5 Y4 h0 W$ F5 d+ w9 y

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

) A: Y7 O, o' s; [' V# J8 E

// Helpfile:

/ ~6 r% v' m+ e" V5 s/ {+ p

// DepndLst:

5 y+ E U8 W/ {

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

* E; `7 L. _0 ~

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

* b, v5 l0 M6 \8 B9 @+ }' q

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

* \- Z; q8 E( g* I# j

#ifndef __PCOMServer_TLB_h__

9 o! W2 B" x) C/ [! G0 ~

#define __PCOMServer_TLB_h__

: @4 b& V7 X: u

#pragma option push -b -w-inl

( [' X- B% b$ _: `5 x$ O

#include <vcl/utilcls.h>

# a9 C x/ a" j' z

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

8 V, B1 F; ^# _* L. l

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

* s' t# S# _ Y; b1 Y1 _

#endif

* }# U! \3 L5 ?8 P

' e: z) r7 e: ~, a

#include <olectl.h>

7 c% E4 A6 F* } [

#include <ocidl.h>

, d) v, j- {# V! J f1 Y) s9 t# f! o

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

( W" C; b2 Y, {8 ^9 r, t

#if !defined(__TLB_NO_EVENT_WRAPPERS)

+ J* b3 o2 F" B' k: M2 y. W( p

#include <atl/atlmod.h>

6 b! V4 s, i z7 e# |# `

#endif

3 S' M' i. ]# `4 L5 N

#endif

- k# M$ y" q) K# E, Q

1 g1 ]7 R0 n0 f# W6 \. c- I i! \8 F

namespace Stdvcl {class IStrings; class IStringsDisp;}

6 d3 i7 j8 d2 z* |' i: |

using namespace Stdvcl;

9 W7 w' |: ]# c

6 A: m6 d- S b8 }2 G' }

namespace Pcomserver_tlb

* F7 \ O. u& X7 h

{

& ]1 X5 F5 {5 Q

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

, }( U/ ^: C# y; ]. P% M8 M

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

7 B6 J9 I5 _1 Z

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

: K* H0 v$ U0 @6 y

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

/ s7 \' J0 u0 F+ x! @

typedef IMyCOM MyCOM;

1 x. T! g8 D$ E F

1 O9 F) B# ~" J+ b d5 x2 ~2 a. v

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

, t. J" n. b+ G$ h7 L) v0 }

interface IMyCOM : public IDispatch

, N7 Z' ~: q0 e

{

: H) D$ R6 R6 G! P# P* K [

public:

( i% u9 g. O7 M3 u9 R, `. ?1 u8 u4 d

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

" U# |3 |5 U$ _6 G

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

' z* {4 C- J! ]

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

% F9 V/ C: @ T' }1 C6 o

{

# B/ x1 }9 N$ E4 Z+ H

int ret;

3 c1 L: o! \' H

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

) A) ~( y/ {8 N9 i

return ret;

- G9 S# f4 P; }( ]& g, z

}

3 x+ ^9 H1 [1 n* g/ ^3 ?1 I

#endif // __TLB_NO_INTERFACE_WRAPPERS

8 V0 R$ d+ J, _/ x4 v% R3 `

};

8 K1 s7 H" N7 I3 B) G

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

# P! k2 \' `; V: B/ ]4 W6 k9 P

template <class T /* IMyCOM */ >

& L& i- a' i* j* w! \5 o

9 z( z' D9 T' {9 n; [

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

2 W9 v9 W2 _5 g0 H" u

{

0 J* u& m8 `" h; R: b# h

public:

7 k+ L3 ], x9 x5 U1 b

TCOMIMyCOMT() {}

: }" K; @, _. T

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

5 J( P9 v0 E, m3 O) u

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

# w+ K2 ~: f6 L+ c8 d% B8 Z

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

5 C9 x/ [& ?4 A7 X; l* G& A9 L

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

2 {1 D+ D/ m5 n2 q

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

2 O8 i7 @# D% L% ?

};

& t. x1 B& D; {4 X7 G. D, Y

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

2 J2 ~: S. ~4 V: ?6 P6 d' O

- E7 G7 o1 D/ [6 ?

template<class T>

5 P: i6 F( Y# n2 D- |$ B0 b

class IMyCOMDispT : public TAutoDriver<IMyCOM>

: R% k' u) v4 L x2 R2 N4 U/ m

{

; h* H, j: U8 a, h

public:

4 k$ G6 o% W7 F1 P$ {8 ?

IMyCOMDispT(){}

8 w! z! b4 ^3 [3 ]7 N4 ^% x

IMyCOMDispT(IMyCOM *pintf)

) V) v `- i2 l0 m/ b1 V

{

1 U; G2 d0 C5 P7 G( y6 U$ @

TAutoDriver<IMyCOM>::Bind(pintf);

# ]9 @, |+ u8 @8 _/ B

}

% L1 ?; k( ^$ x) l' v

IMyCOMDispT& operator=(IMyCOM *pintf)

& l9 m, a9 F, \3 D! s H

{

8 K- G& v/ X" G! U' l

TAutoDriver<IMyCOM>::Bind(pintf);

! L; ]% H0 E. S2 f9 F

return *this;

. j# W( S/ p" p! b2 H

}

0 v* q2 o' K2 G2 [; b1 R0 K

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

* q+ B* z; q' b$ {

{

) q4 b- l- d" z& i

return OLECHECK(Bind(CLSID_MyCOM));

0 @* Y1 g8 m9 ~( x( O0 h7 ]

}

; g9 R, {' N/ D3 d- S0 e

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

, d! z* e+ U. _0 f

{

. }# j# {; p, L" U& N7 @ P: o/ v l* }

return BindToActive(CLSID_MyCOM);

0 R& i% R) q i) w" p) h

}

1 H+ Y2 p8 w Q- o

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

2 a9 \8 d. W4 S7 R/ s

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

- l4 }+ e" ?% z) |4 l Y

};

8 o" b) S# v! e* C5 q/ |( J

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

8 K9 E, {: V: _* M7 K$ q8 `

9 q" |* {( V: \' Y

template <class T> HRESULT __fastcall

" g) H. w- ]" F

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

7 J7 Q% j& Q4 J5 ~: m m

{

" v2 e* i8 Z. `3 n1 p4 J

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

- D- Q7 }2 i# W

}

' }' O1 @2 w) X: o9 D

3 Q" Z+ {9 H. K

template <class T> int __fastcall

3 d3 s2 k4 v9 h

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

$ F2 A) w$ S5 S' J: q% J3 ?" k

{

7 J- o5 w& a. p% @! S L# H& r0 y

int ret;

% K( y/ B- e* X% ]5 F

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

% ]3 @5 R4 z" j' R7 Z' R

return ret;

( w o6 O J! `6 u

}

& k! t1 h i: N1 L. d

( g! l* w/ c& c- p6 W$ x5 z* l* |

template <class T> HRESULT __fastcall

. f% I2 G& K9 e/ M$ r

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

- T- l/ `! _9 ]4 p' m7 ?

{

9 P4 W$ Q7 K0 p+ _

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

4 ~; `7 ^( @5 K6 g- g

TAutoArgs<2> _args;

; {0 O4 n0 I7 e' `/ k0 A

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

1 ~ t% g8 A S8 s

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

: K1 j9 Q8 W; {6 {4 F

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

3 l; s |% C" @" B/ @

}

9 F( e: r! w( ?; ~# N! k) B; i, @

$ l! {" E8 t$ D; H

template <class T> int __fastcall

7 ^+ x5 X6 S! Y/ r9 P+ W/ b

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

* D* q5 y( X5 A" x, d! r

{

. }" k6 ?) {" Z$ U

int ret;

$ M; i2 P. I% H1 c- Y$ ^% V

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

# K* z! N T, c. U% f3 z' U. f

return ret;

# Z! q' O4 |6 f# g

}

; k7 t) ?5 f9 L" j1 {

7 G/ m( a5 ]& K& Y& {, f2 \. n5 P1 j

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

; L `8 v4 y9 @0 D, L

#endif // __TLB_NO_INTERFACE_WRAPPERS

6 q6 e% ~# h s8 _3 N! S( R

}; // namespace Pcomserver_tlb

* c2 D0 o C" n+ H' u! Z

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

4 G/ K' r' i) p7 G" B

using namespace Pcomserver_tlb;

: K; a% t1 O( s* L) t9 K

#endif

: J, D# Y" h2 d6 o4 {- C5 i

#pragma option pop

) N, P2 A$ m H

#endif // __PCOMServer_TLB_h__

. B; g4 {6 o/ J" [, @

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

+ t" J0 ? O- d& z* s

interface IMyCOM : public IDispatch

: C: T4 x- o- }$ }* t( V7 @

class TCOMIMyCOMT : public TComInterface<IMyCOM>

* n* c7 J& {* h4 L1 g

class IMyCOMDispT : public TAutoDriver<IMyCOM>

: D' V( P8 F n" `

class CoMyCOM: public CoClassCreator

9 q, v+ h6 F0 q! K/ f% s

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

( t0 @1 J" F4 `* ~. h1 e

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

( w( g) j/ L+ C n: ^( Q1 n& i1 C

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

# s6 X4 Z3 y' R; ?" k

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

7 W2 Z: z% f& ?7 I+ C

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

. Y, N, d7 @8 }, ?3 v) A

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

, D6 S# E5 L4 Q

int x=6,y=6;

% z6 ]1 y+ i! a/ C. ^6 Y

TCOMIMyCOM O;

$ b5 j j6 f1 ?0 C5 L6 o% [

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

- a7 ?, T1 o9 B& l U, z( ^% M5 M; t

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

! j/ p% c) W7 ]! M$ }; j' S6 t2 `

Edit1->Text=IntToStr(y);

1 P2 k1 w8 M6 X z( H; s

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

& V- Y# p6 B4 b+ E- `+ [8 R. }" H

int x=6,y=6;

1 y2 w) S& ]5 Z n* A# N" E; R F

IMyCOMDisp app;

8 a6 ~& m1 j: f& [

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

! H. N! k# K9 P$ F( x

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

$ I# \' ]! S" F, ~0 g

Edit1->Text=IntToStr(y);

0 a$ b8 V: u0 l2 o' Q3 ?% i

4.小结

! B- p$ [* C, T5 Q

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

# q: D) T* S* u: H) |0 e

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

. y" Y% |( K$ {) w) B2 j

/ s4 Z; M+ c! W1 O+ n

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 10:44 , Processed in 0.381131 second(s), 52 queries .

回顶部