假设需求如下:底层是一个数学运算库DLL,中间是ActiveX控件(它调用底层的数学运算库DLL来完成控制层),界面层在测试时可以是一个exe程序,最后发布到IE浏览器上测试。

 

数学运算库DLL的开发

      新建一个Win32 DLL项目,加入一个头文件MyNum.h,在其中声明所有的数学函数(为简单起见,本文只考虑加法运算),代码如下:

COM组件开发实践(二)_控件#ifndef MY_NUM_H
COM组件开发实践(二)_控件
#define MY_NUM_H
COM组件开发实践(二)_控件
int __stdcall AddNum(int,int);
COM组件开发实践(二)_控件
#endif
COM组件开发实践(二)_控件

请注意这里的方法声明为__stdcall,而VC++默认的是__cdecl,由于组件的语言无关性要求调用和被调双方必须在函数调用的约定上一致,因此在后面加载DLL并获取此方法时也要求和你的声明一致。

      为了简单起见,加法方法的实现就放倒DLL入口点所在文件,代码如下:

COM组件开发实践(二)_控件// NumDLL.cpp : 定义DLL 应用程序的入口点。
COM组件开发实践(二)_控件
//
COM组件开发实践(二)_控件
#include "stdafx.h"
COM组件开发实践(二)_控件#include 
"MyNum.h"
COM组件开发实践(二)_控件
COM组件开发实践(二)_控件#ifdef _MANAGED
COM组件开发实践(二)_控件
#pragma managed(push, off)
COM组件开发实践(二)_控件
#endif
COM组件开发实践(二)_控件
COM组件开发实践(二)_控件
int __stdcall AddNum(int Num1,int Num2)
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19    
return Num1+Num2;
COM组件开发实践(二)_html页面_20}

COM组件开发实践(二)_控件BOOL APIENTRY DllMain( HMODULE hModule,
COM组件开发实践(二)_控件                       DWORD  ul_reason_for_call,
COM组件开发实践(二)_控件                       LPVOID lpReserved
COM组件开发实践(二)_控件                     )
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19    
return TRUE;
COM组件开发实践(二)_html页面_20}

COM组件开发实践(二)_控件
COM组件开发实践(二)_控件#ifdef _MANAGED
COM组件开发实践(二)_控件
#pragma managed(pop)
COM组件开发实践(二)_控件
#endif
COM组件开发实践(二)_控件

      为了能在其他程序中显示链接此DLL,我们为它加入一个.def文件,命名为NumDLL.def,列出此DLL导出的方法名称:

COM组件开发实践(二)_控件LIBRARY    "NumDLL"
COM组件开发实践(二)_控件EXPORTS
COM组件开发实践(二)_控件    AddNum
COM组件开发实践(二)_控件

至此我们的数学运算函数库DLL就完成了。

ATL开发ActiveX控件

      开发ActiveX控件有两种方式,一是MFC,二是ATL,而后者是专门用于COM组件开发,因此更适合于ActiveX。因此这里选择后者,前者的开发示例参考我这篇文章(用VC++开发ActiveX 控件完全教程(一))。

      新建一个ATL项目,命名为”FuckATL”,接受默认设置。右键项目名,添加一个”ATL简单对象,命名为CaluNumCtrl点击下一步进入组件选项设置界面。

      修改类的头文件CaluNumCtrl.h如下:

COM组件开发实践(二)_控件class ATL_NO_VTABLE CCaluNumCtrl :
COM组件开发实践(二)_控件    
public CComObjectRootEx<CComSingleThreadModel>,
COM组件开发实践(二)_控件    
public CComCoClass<CCaluNumCtrl, &CLSID_CaluNumCtrl>,
COM组件开发实践(二)_控件    
public ISupportErrorInfo,
COM组件开发实践(二)_控件    
public IConnectionPointContainerImpl<CCaluNumCtrl>,
COM组件开发实践(二)_控件    
public CProxy_ICaluNumCtrlEvents<CCaluNumCtrl>,
COM组件开发实践(二)_控件    
public IObjectWithSiteImpl<CCaluNumCtrl>,
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17    
public IDispatchImpl<ICaluNumCtrl, &IID_ICaluNumCtrl, &LIBID_FuckATLLib, /**//*wMajor =*/ 1/**//*wMinor =*/ 0>
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19
public:
COM组件开发实践(二)_html页面_19    typedef 
int (__stdcall*PtrAddNum)(int,int);
COM组件开发实践(二)_html页面_19    PtrAddNum MyAddNum;
COM组件开发实践(二)_html页面_19    CCaluNumCtrl()
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56    
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19        
//加载数学运算库DLL
COM组件开发实践(二)_html页面_19
        handle = ::LoadLibrary(_T("D:\\dyk\\work\\NumDLL\\debug\\NumDLL.dll"));
COM组件开发实践(二)_html页面_19        
if (handle == NULL) 
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56        
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19            DWORD e 
= GetLastError();
COM组件开发实践(二)_html页面_19            
return;
COM组件开发实践(二)_#endif_66        }

COM组件开发实践(二)_html页面_19        
//获取加法运算函数指针
COM组件开发实践(二)_html页面_19
        MyAddNum = (PtrAddNum)GetProcAddress(handle,"AddNum");
COM组件开发实践(二)_#endif_66    }

COM组件开发实践(二)_html页面_19DECLARE_REGISTRY_RESOURCEID(IDR_CALUNUMCTRL)
COM组件开发实践(二)_html页面_19BEGIN_COM_MAP(CCaluNumCtrl)
COM组件开发实践(二)_html页面_19    COM_INTERFACE_ENTRY(ICaluNumCtrl)
COM组件开发实践(二)_html页面_19    COM_INTERFACE_ENTRY(IDispatch)
COM组件开发实践(二)_html页面_19    COM_INTERFACE_ENTRY(ISupportErrorInfo)
COM组件开发实践(二)_html页面_19    COM_INTERFACE_ENTRY(IConnectionPointContainer)
COM组件开发实践(二)_html页面_19    COM_INTERFACE_ENTRY(IObjectWithSite)
COM组件开发实践(二)_html页面_19END_COM_MAP()
COM组件开发实践(二)_html页面_19
COM组件开发实践(二)_html页面_19BEGIN_CONNECTION_POINT_MAP(CCaluNumCtrl)
COM组件开发实践(二)_html页面_19    CONNECTION_POINT_ENTRY(__uuidof(_ICaluNumCtrlEvents))
COM组件开发实践(二)_html页面_19END_CONNECTION_POINT_MAP()
COM组件开发实践(二)_html页面_19
// ISupportsErrorInfo
COM组件开发实践(二)_html页面_19
    STDMETHOD(InterfaceSupportsErrorInfo)(REFIID riid);
COM组件开发实践(二)_html页面_19    DECLARE_PROTECT_FINAL_CONSTRUCT()
COM组件开发实践(二)_html页面_19    HRESULT FinalConstruct()
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56    
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19        
return S_OK;
COM组件开发实践(二)_#endif_66    }

COM组件开发实践(二)_html页面_19    
void FinalRelease()
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56    
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_#endif_66    }

COM组件开发实践(二)_html页面_19
//组件对外放出的加法方法
COM组件开发实践(二)_html页面_19
public:
COM组件开发实践(二)_html页面_19    STDMETHOD(AddNumbers)(LONG Num1, LONG Num2, LONG
* ReturnVal);    //下面是一个NUM属性,也是用于测试,包含了读写设置方法
COM组件开发实践(二)_html页面_19
public:
COM组件开发实践(二)_html页面_19    STDMETHOD(get_NUM)(SHORT
* pVal);
COM组件开发实践(二)_html页面_19
public:
COM组件开发实践(二)_html页面_19    STDMETHOD(put_NUM)(SHORT newVal);
COM组件开发实践(二)_html页面_19
public:
COM组件开发实践(二)_html页面_19    HMODULE handle;
//数学函数库模块句柄
COM组件开发实践(二)_html页面_20
}
;
COM组件开发实践(二)_控件

在控件实现文件CaluNumCtrl.cpp中,代码如下:

COM组件开发实践(二)_控件STDMETHODIMP CCaluNumCtrl::AddNumbers(LONG Num1, LONG Num2, LONG* ReturnVal)
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19    
int sum = this->MyAddNum(static_cast<int>(Num1),static_cast<int>(Num2));
COM组件开发实践(二)_html页面_19    
*ReturnVal = static_cast<LONG>(sum);
COM组件开发实践(二)_html页面_19    
return S_OK;
COM组件开发实践(二)_html页面_20}

COM组件开发实践(二)_控件
COM组件开发实践(二)_控件STDMETHODIMP CCaluNumCtrl::get_NUM(SHORT
* pVal)
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19    
*pVal = 10;
COM组件开发实践(二)_html页面_19    
return S_OK;
COM组件开发实践(二)_html页面_20}

COM组件开发实践(二)_控件
COM组件开发实践(二)_控件STDMETHODIMP CCaluNumCtrl::put_NUM(SHORT newVal)
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19    
// TODO: 在此添加实现代码
COM组件开发实践(二)_html页面_19
    
COM组件开发实践(二)_html页面_19    
return S_OK;
COM组件开发实践(二)_html页面_20}

COM组件开发实践(二)_控件

     好了,ActiveX控件仅仅是简单地调用底层的数学运算库DLL来完成运算,下面我们写一个exe程序对这个COM组件进行测试。

一个控制台测试程序

      建立一个最简单的控制台程序来进行测试,代码如下:

COM组件开发实践(二)_控件#include "..\..\FuckATL\FuckATL\FuckATL.h"
COM组件开发实践(二)_控件#include 
"..\..\FuckATL\FuckATL\FuckATL_i.c"
COM组件开发实践(二)_控件#include 
<iostream>
COM组件开发实践(二)_控件
using  namespace std;
COM组件开发实践(二)_控件
COM组件开发实践(二)_控件
void main(void)
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19    
// Declare and HRESULT and a pointer to the Simple_ATL interface
COM组件开发实践(二)_html页面_19
    HRESULT            hr;
COM组件开发实践(二)_html页面_19    ICaluNumCtrl 
*IFirstATL = NULL;
COM组件开发实践(二)_html页面_19    
// Now we will intilize COM
COM组件开发实践(二)_html页面_19
    hr = CoInitialize(0);
COM组件开发实践(二)_html页面_19    
// Use the SUCCEDED macro and see if we can get a pointer to
COM组件开发实践(二)_html页面_19    
// the interface
COM组件开发实践(二)_html页面_19
    if(SUCCEEDED(hr))
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56    
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19        hr 
= CoCreateInstance( CLSID_CaluNumCtrl, NULL, CLSCTX_INPROC_SERVER,
COM组件开发实践(二)_html页面_19            IID_ICaluNumCtrl, (
void**&IFirstATL);
COM组件开发实践(二)_html页面_19        
// If we succeeded then call the AddNumbers method, if it failed
COM组件开发实践(二)_html页面_19        
// then display an appropriate message to the user.
COM组件开发实践(二)_html页面_19
        if(SUCCEEDED(hr))
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56        
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19            
long ReturnValue;
COM组件开发实践(二)_html页面_19            hr 
= IFirstATL->AddNumbers(57&ReturnValue);
COM组件开发实践(二)_html页面_19            cout 
<< "The answer for 5 + 7 is: " << ReturnValue << endl;
COM组件开发实践(二)_html页面_19            
short num; 
COM组件开发实践(二)_html页面_19            IFirstATL
->get_NUM(&num);
COM组件开发实践(二)_html页面_19            cout
<<"num is: "<<num<<endl;
COM组件开发实践(二)_html页面_19            hr 
= IFirstATL->Release();
COM组件开发实践(二)_#endif_66        }

COM组件开发实践(二)_html页面_19        
else
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56        
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19            cout 
<< "CoCreateInstance Failed." << endl;
COM组件开发实践(二)_#endif_66        }

COM组件开发实践(二)_#endif_66    }

COM组件开发实践(二)_html页面_19    
// Uninitialize COM
COM组件开发实践(二)_html页面_19
    CoUninitialize();
COM组件开发实践(二)_html页面_19    system(
"pause");
COM组件开发实践(二)_html页面_20}

COM组件开发实践(二)_控件

来到IE的世界

      最后我们将此ActiveX组件嵌入到html页面中,对其进行测试.新建一个html页面,代码如下:

COM组件开发实践(二)_控件<HTML>
COM组件开发实践(二)_控件
<HEAD>
COM组件开发实践(二)_控件
<TITLE>New Page</TITLE>
COM组件开发实践(二)_#include_16COM组件开发实践(二)_#pragma_17
<script language="javascript">COM组件开发实践(二)_html页面_18
COM组件开发实践(二)_html页面_19    
function doTest()
COM组件开发实践(二)_#include_55COM组件开发实践(二)_#include_56    
COM组件开发实践(二)_html页面_18{
COM组件开发实践(二)_html页面_19        
var sum = FuckATL1.AddNumbers(3,4);
COM组件开发实践(二)_html页面_19        alert(sum);
COM组件开发实践(二)_html页面_20    }

COM组件开发实践(二)_控件
</script>
COM组件开发实践(二)_控件
</HEAD>
COM组件开发实践(二)_控件
<BODY>
COM组件开发实践(二)_控件
<OBJECT ID="FuckATL1" CLASSID="CLSID:7BF3B65F-A800-4604-AE6B-91844EFD5F05">
COM组件开发实践(二)_控件
</OBJECT>
COM组件开发实践(二)_控件
<input type="button" value="测试加法" id="btnOK" onclick="doTest();"></input>
COM组件开发实践(二)_控件
</BODY>
COM组件开发实践(二)_控件
</HTML>
COM组件开发实践(二)_控件

     由于暂时先不考虑控件的安全性需要,因此会出现下面的警告信息,不过不要紧,这个问题以后再解决。

COM组件开发实践(二)_#endif_203

测试结果如下:

COM组件开发实践(二)_html页面_204