QQ登录

只需要一步,快速开始

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

COM应用软件开发技术

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

823

主题

3

听众

4048

积分

我的地盘我做主

该用户从未签到

发帖功臣 元老勋章

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

蔡倩

1 P9 T, }! k w2 K/ |/ F

主题词: COM ActiveX C++ Builder

( R0 @1 r" l. }4 {

1.COM技术概述

; c, @8 p6 t! ]

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

' ]# X9 C) \1 f4 B! a- h

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

) U: ^9 Y, ~6 k4 F3 e9 Z

◆用于创建ActiveX控件。

: J9 x- O1 B l# K

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

) ^; @; V! e8 t. W; I. m

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

- w- i( x4 Y% l( a# |9 ]+ v

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

( H* C3 E& j& d- R

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

# [4 R3 M4 \( E3 o

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

- Q- D: I' {( }

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

9 D$ ]# `( `# k6 I4 \6 f1 p. q

2.建立COM服务程序

# `8 ^& @; x6 A( [3 A' h

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

! I0 l. n' l: L7 p

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

1 o9 D7 G2 i+ ~& e

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

8 `9 @! f+ i; W

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

; m* ]; L5 Z. z5 k( Y

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

$ U, `( u: g/ Z9 }- |* N

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

$ c' e3 J0 j, M1 y; T5 p

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

3 \ ?8 K3 n2 |. w4 A

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

% K/ q0 s1 x6 r, K

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

b9 w3 h: Z2 L

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

7 p: F( e) l' i. T/ [

2.2建立COM接口对象

( N& r/ `8 l, C

' U) \7 z! ~, e7 R# G/ t

- ^; M( _; ]3 }( {7 V

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

. ?! y' k# s. n- P* S

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

2 A% W4 V9 x4 \" a% B+ U( e

8 I: c1 Z+ i8 }* }7 e3 C' ]* ]7 V

4 \! }+ \: u! w' ?

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

/ J) K6 i i# I [& C0 O9 h! |, b

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

9 V/ h% z( m" @

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

) z7 [( i( D. @8 d3 j

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

, W7 Q6 ^4 m! E

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

8 [# H0 j0 E( C! n- w

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

6 s. n! a/ p& ]* _3 M

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

# r3 @8 S# q* C( s8 p

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

0 A! I! b$ ~4 a n6 d. B. C

[id(0x00000001)]

# N4 t/ p! B V% \6 k o0 E

- d# Y& G% G1 `6 S( V* _3 F

1 J6 a* P, Q! }2 a

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

" j( S9 q0 M; L: o X3 E: @

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

7 C2 R4 q- C& Z( F

PCOMServer.TLB: 类型库文件;

; {! `$ f) a! [5 U8 o

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

' f" S: j5 j" W! {, q

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

8 J @7 O7 h J# @% M. Z2 ^

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

4 o5 _& @$ \/ T

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

, J9 o& Y% u k* q9 Y$ ^1 W, J

2.4 实现COM接口中的方法

7 k: ^5 S# J4 g4 }% _

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

) v3 F; i3 j9 d

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

/ v6 E& {# _3 [' @% g

{

V* b: [ J t& A. H: J! Q

*ret=x+y;

( m {. y5 x* D' H+ B& X

return S_OK;

/ U" v/ g* N. f! J' L

}

% U. W$ w% u# _0 s7 z: W

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

3 `8 i2 @6 r% L

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

+ o- S$ P3 v' ?. ]; }. ?( }, E

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

/ P1 b) _6 B) \( F [$ O+ l

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

7 s5 R- A' |; h* I1 |/ q @# ]

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

. E/ M4 o2 J; R! f' N- E# O) z

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

' r5 Z G. e/ N3 V% [

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

2 S! y8 C' X2 o& ~3 C4 Z9 O, s

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

5 W% b/ M1 ]/ L; p" {; V* N# \

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

/ |7 L, G* }5 W# ~/ s9 l

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

, R% x) A' u% q- c% D

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

9 y8 o6 [# r7 u8 a1 o8 S

3.建立COM客户程序

) Y, r/ \2 ^+ i* v# P( _. z- G) J

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

* I, Z, i/ \! f0 v

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

8 r# L0 |( e6 k% Y2 y" c7 X

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

) K% ?* @' B: V8 L; {( K. E

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

. W5 O+ G9 V. w2 L" d1 R

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

" O1 x9 ~5 N2 N8 j! D r' v7 w

// Helpfile:

, x/ m" d9 J7 a6 s+ E) `

// DepndLst:

, N( c/ [$ G6 B% P0 `9 e

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

% A( _$ I! |2 E. U. ^

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

6 K8 ^3 r5 ^/ d

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

* Y! T+ g4 U/ H0 e0 g

#ifndef __PCOMServer_TLB_h__

0 L5 N) U, \+ \

#define __PCOMServer_TLB_h__

, c/ R ?' v7 @' o8 A

#pragma option push -b -w-inl

7 j- ?" s! y$ g0 X, |

#include <vcl/utilcls.h>

2 p+ l. _7 g$ c" v

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

3 n% S4 O! j6 P

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

& K N B; k$ x+ t' M

#endif

# T0 g T/ @1 `- z8 O

& f/ i0 I m7 T: N% v& V

#include <olectl.h>

: I- S9 F1 s/ v7 L+ z

#include <ocidl.h>

2 l% [3 b5 E8 n3 y2 u

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

/ ?8 M- c( y9 ^

#if !defined(__TLB_NO_EVENT_WRAPPERS)

- d, s0 @/ r1 L! K3 h

#include <atl/atlmod.h>

* `, o4 I; v2 @# ?

#endif

3 `1 G& j5 T/ q# y5 b

#endif

" V8 _/ _0 k: j. o/ C

6 @0 }$ _' [8 }7 u, z

namespace Stdvcl {class IStrings; class IStringsDisp;}

5 p5 G6 q/ Z" w; G) O( i$ }# h

using namespace Stdvcl;

! w3 {" h B( G

{5 r( |$ [! ]+ N

namespace Pcomserver_tlb

6 c" m W% F' L. Y4 W8 I$ z

{

, _0 u8 }( b- Y* v

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

$ r8 a# e. j& E2 q. X

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

& E/ z S( ~" b

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

5 F. R& S) `; y5 `3 @! z

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

5 C! c+ x W/ }1 ^6 a# c( z

typedef IMyCOM MyCOM;

2 t, ?! @7 t C9 z

5 o8 K6 g3 @7 p9 M# ], m2 P+ o9 _

#define LIBID_OF_MyCOM (&LIBID_PCOMServer)

4 E# p: r0 v$ r0 U8 V' e+ G+ p

interface IMyCOM : public IDispatch

/ z* P9 y$ S% R7 ?+ ~* K

{

4 K" e5 U- S0 A: M5 m' o

public:

* c+ d6 F( x7 t r; i4 ^

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

5 f. p" Z7 Q4 E: Z0 G$ q

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

3 U( L0 _7 y' R( i+ R

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

" h$ w1 ?1 }9 o

{

5 u7 Z4 U! ~5 d; W

int ret;

/ G6 y* Y0 w3 F

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

) C2 \, P- P0 I# ]

return ret;

) N" K+ L, h" ]4 k0 e- T9 s

}

1 j/ K9 F) W+ v7 f& }$ d

#endif // __TLB_NO_INTERFACE_WRAPPERS

- ? Q- m) J7 R

};

6 `9 N5 B3 v( T# J* ^# r! u

#if !defined(__TLB_NO_INTERFACE_WRAPPERS)

; W4 }) I5 O; \0 C, g4 |

template <class T /* IMyCOM */ >

. ?2 G, s7 _! b) ~# A9 f

- i7 F8 \* U$ E6 Z4 k

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

6 x( O9 ]- `0 V4 G1 Y2 L

{

( V7 U, n& a" t. u, i$ _ d

public:

; a6 O" y9 H/ b g) i# g) _% e

TCOMIMyCOMT() {}

8 _$ }' g3 Q* H* Y$ y) ?

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

8 }0 h7 b" t# [7 ?3 f2 V. v

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

1 b: l. h% F, P4 v+ e5 [

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

. }- n# e9 `1 y

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

2 p1 G X4 B7 p$ y0 F

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

2 E$ P3 _2 N8 H2 U5 ?4 r

};

" V9 x" Z8 q) _5 l, I/ V2 i

typedef TCOMIMyCOMT<IMyCOM> TCOMIMyCOM;

* W& t& J0 T7 Q! e4 N( J. _

6 t$ q4 v6 \# X' a+ @

template<class T>

) _+ _4 b5 ?) X2 a( X

class IMyCOMDispT : public TAutoDriver<IMyCOM>

% P: Y+ u4 D& |0 N

{

' }# u% V7 J5 a# t4 }/ u

public:

( K$ g% _/ a' V5 p! I, P/ E

IMyCOMDispT(){}

/ p7 y0 }1 n; T+ s9 a X& B

IMyCOMDispT(IMyCOM *pintf)

8 ?* ~6 u7 _ p

{

$ ^3 S# P* l4 z7 G- M3 a5 I3 E' V

TAutoDriver<IMyCOM>::Bind(pintf);

2 m7 n6 k8 [. z* v; _

}

( \/ [9 k9 l$ `. J1 P

IMyCOMDispT& operator=(IMyCOM *pintf)

; K# _/ p5 U! G* G7 s2 Z, x

{

$ n* q& G7 T/ b0 l6 m- u4 o! ?2 B

TAutoDriver<IMyCOM>::Bind(pintf);

) b; t$ g- {2 y

return *this;

E# l3 }) P" ~4 ^

}

& r; j8 d, I4 z9 Q8 k

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

6 W/ M" Q s/ H* a6 B- b5 ]

{

0 E3 O) s! O+ z/ V8 q+ o

return OLECHECK(Bind(CLSID_MyCOM));

$ T! E. p/ g. q9 Q* G

}

! C" s5 f- p; R+ E/ X, }

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

/ [: a" Q7 j/ P. I3 r

{

, j- W3 v3 M" M1 \; I

return BindToActive(CLSID_MyCOM);

$ n: @% q. d/ T( m, F

}

1 ? u2 W- g6 M7 Y- }

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

, y( \2 E1 } p$ T

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

: S. w1 O4 L( i# `( U7 J, K

};

- K6 r m8 |- K9 l9 f# y

typedef IMyCOMDispT<IMyCOM> IMyCOMDisp;

* P8 ]# W5 p! p2 O

8 }2 @6 k) ]8 u$ ], e

template <class T> HRESULT __fastcall

8 {- i( n- b) T; q1 A0 {* ?' M

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

% g. x. d q8 X, L; J6 ~ Y

{

~- G3 e8 Z) F: x8 x& T$ F' ~7 c/ C; Q' G

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

; a' }. T# h3 t- Q

}

6 i+ A) b# R/ [# n/ L- G3 ^

! P: K& I; s) g) N7 S

template <class T> int __fastcall

' P/ q! m3 k7 [: R( Q: u m! Z

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

! u/ k8 b3 z8 a( D" s

{

0 n, ?, ?" e2 r

int ret;

9 ~) R2 j* f9 ~" w

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

: D' t' `- u5 u

return ret;

5 H6 ?/ X5 S6 Y# }

}

8 C! K% d# b; ^9 q2 B; m

. o7 p3 C/ N. z1 _5 ~' j

template <class T> HRESULT __fastcall

- Q/ e; ]1 z3 F7 y% |' |( l

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

) |" c, E. |6 C

{

' h: k" x5 c% Z2 u1 |+ n2 b% n4 L

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

+ ^0 d3 E7 a( o/ L9 D$ j

TAutoArgs<2> _args;

8 |. a2 C7 X" [6 r" T6 i3 U

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

" D, R& s- e2 \( T4 o

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

4 n: ~$ Q4 Q% E t! }) d8 x8 W; L

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

3 P8 c! F2 c8 p2 w0 X6 `" \; e1 `* a

}

# D3 _- \/ F) K. d' d7 O

! X6 j- I* }# m, h. r+ i+ W: u

template <class T> int __fastcall

8 t6 j# d1 t2 k# U+ H# X

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

6 @. x6 w' n" a

{

7 q1 }7 ^! k8 r; F6 g- P

int ret;

9 W: \# @& b: s$ k( m6 C

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

8 L. q5 Q+ O! A) t# G8 p* t$ X D+ h

return ret;

' p! Z+ N; s: o* h1 {3 h6 l

}

. R( {/ `9 x; C* o9 g8 H& _5 G" \

( X' ?0 m9 {0 S1 t4 x% c6 m

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

. Q6 g m3 t) F) c5 r* m

#endif // __TLB_NO_INTERFACE_WRAPPERS

a, ~$ x5 T# J3 b& ^

}; // namespace Pcomserver_tlb

6 r/ E4 y9 h& |; z$ d1 z* x$ U& }

#if !defined(NO_IMPLICIT_NAMESPACE_USE)

6 w0 }2 U* g. f0 Q

using namespace Pcomserver_tlb;

6 O6 ?1 a5 i. }- n

#endif

# e4 G9 K3 t" b! x

#pragma option pop

' v. g$ }$ @) c* ?

#endif // __PCOMServer_TLB_h__

% O$ n& g% a( C$ C8 z* L

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

; O a+ W9 J+ x

interface IMyCOM : public IDispatch

v- V4 t X& ~, v3 Q

class TCOMIMyCOMT : public TComInterface<IMyCOM>

- l; Z& i% U" {5 U- i; U$ k

class IMyCOMDispT : public TAutoDriver<IMyCOM>

: ]1 p% Z7 j! y0 k3 R" C8 V

class CoMyCOM: public CoClassCreator

; U8 W6 V; ]# j. S/ I$ O

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

. I( r: k) b0 h) o0 J

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

" D) Y0 o1 i+ M: ]

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

, \8 i& W' W$ q' N5 b" g0 `

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

- t2 P" O; p4 h- H; [

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

/ {" U7 {/ w" P5 {

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

+ f6 t, p4 ]* w& W4 I. t; m

int x=6,y=6;

) [" f3 _0 T' n. I4 T* N

TCOMIMyCOM O;

9 \- \1 k- I' h/ S7 M4 h

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

* a: Z: y# _4 U- i1 A

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

. K& |& `. q l( R1 @6 z6 A, s

Edit1->Text=IntToStr(y);

( ^- N8 Z) d6 g3 u! {% n* A% N

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

4 n: S, N- u# O) x' O6 F( Z/ R

int x=6,y=6;

& [/ a, v0 M+ t) S! l7 j

IMyCOMDisp app;

7 l1 g+ w& o: n f/ z

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

# y! H7 l; h; d

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

; \' m) u( Y( E2 R. _

Edit1->Text=IntToStr(y);

0 E$ e! K5 _1 o4 B. D8 v1 I

4.小结

6 m! e+ v. U" Z

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

' e2 d8 [ D$ Z7 a+ {6 g4 Z. y

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

& d( m! G: [# @ T# }# e

6 w( ^$ G' e. b. v, r% A

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, 2025-12-28 16:52 , Processed in 0.542291 second(s), 52 queries .

回顶部