我原来以前写dll创建过程,而直接使用LoadLibrary加载动态库。
但ATL提出了一个非常重要的特点是引入COM对象的概念。
首先。 ATL active template library该活动模板库。ATL至asp提供代码COM应用。
而一般使用活动模版库来创建COM组件。简单来说ATL一般作为方便快捷的COM开发工具使用。
而ATL中使用的基本技术为 COM技术。C++模版技术和C++多继承技术。
接下来是使用ATL开发一个COM组件的基本过程。附带截图:
1.新建一个ATL项目:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
随便取个名字,然后确定。
设置向导里直接选完毕,这里就是简单的创建一个dll文件。
2. 然后是在项目中选择加入一个新的类:
新类选择为ATL中的ATL简单对象:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
3.在向导中加入类名:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
会发现。向导会自己主动生成其它信息,如类名。组件类名。接口名等。
在选项中还有更详细的一些设置:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
直接点完毕。
4.在类视图中,我们看到有生成的Cfirst类和Ifirst 接口。
在接口上加入方法。会有下面具体的信息:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
填上方法名,然后加入參数。com函数的返回值都是用来检測dll函数是否正确载入并执行的。所以要使用參数作为返回值来传递。这里实现一个简单的加法函数。对于这个函数,其參数为 A,B,和Ret。ret是一个long* 。用来储存返回值。
点in。说明这个參数是是输入的參数。在生成的代码处会自己主动加入_in_ ,提示这里是输入的參数,但没有太大的实际作用。在參数类型中寻找相应的数据类型,并填上相应的參数名:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
对于函数没有返回值的话,通常会将最后一个參数设置为指针类型。用其记录函数返回值:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
最后的參数为:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbHVvX3hpYW5taW5n/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" >
点完毕。
5.改动生成的代码。加入实现。
首先我们双击接口,会进入一个MyFirstDll.idl 的文件里,当中有下面代码:
interface Ifirst : IUnknown{ [] HRESULT Add([in] LONG A, [in] LONG B, [out,retval] LONG* Ret); }; [ uuid(F7DAFD6A-C1DB-46AF-9CF9-62EC0D7D589F), version(1.0), ] library MyFirstDllLib { importlib("stdole2.tlb"); [ uuid(D2F7F834-D4A2-4AB7-B5BE-B08D1EB35564) ] coclass first { [default] interface Ifirst; }; };
是供其它语言或系统中使用的接口。
这里的IUnknown也是值得注意的地方,在COM的内部实现中。全部的类的是继承与这个IUnKnown基类。
对于ATL COM中的一些更加深入的细节。包含:(IUnKnown三个函数,另外两个类CComCoClass . CComObjectRootEx 。DLL的注冊与反注冊。对象的析构函数以及内存申请与释放。首先 com对象中使用new或 CoTaskMemAlloc申请内存都能够,前者是内部自己使用。自己释放。后者可能申请的内存给对方使用,所以对方能够同过相应的方法来释放内存。)这些内容以后有空再更加具体的去学习吧,先挖个坑。以后有空再填。
然后看 first.h。 头文件里关于Cfirst类的声明例如以下:
class ATL_NO_VTABLE Cfirst : public CComObjectRootEx<CComSingleThreadModel>, public CComCoClass<Cfirst, &CLSID_first>, public Ifirst { public: Cfirst() { } DECLARE_REGISTRY_RESOURCEID(IDR_FIRST) DECLARE_NOT_AGGREGATABLE(Cfirst) BEGIN_COM_MAP(Cfirst) COM_INTERFACE_ENTRY(Ifirst) END_COM_MAP() DECLARE_PROTECT_FINAL_CONSTRUCT() HRESULT FinalConstruct() { return S_OK; } void FinalRelease() { } public: STDMETHOD(Add)(LONG A, LONG B, LONG* Ret); };
这里就体现了ATL的基本技术,多继承以及模版。继承 的 内容 有一个 Ifirst , 为之前在idl文件里的接口。在编译器中,对于这个Ifirst接口直接是红下划线表示错误,说明事实上现是一个动态的过程,至于详细是怎样的。我就不想太深入了解了。
这个类声明中有很多复杂的宏的使用。这不是重点,重点是public 中的add函数声明。
然后我们进入 Cfirst.cpp中。在add函数中加上简单的实现:
STDMETHODIMP Cfirst::Add(LONG A, LONG B, LONG* Ret) { // TODO: 在此加入实现代码 *Ret = A+B; return S_OK; }
这样就实现了一个简单的COM组件。
6. 点生成,生成dll。
之后是转让COM物。
版权声明:本文博主原创文章,博客,未经同意不得转载。