DLL+ ActiveX控件+WEB页面调用例子
一、 概述 因項(xiàng)目需要,開始學(xué)習(xí)并研究VC、DLL及ActiveX控件,網(wǎng)上資料找了很多,但沒一個(gè)可用的或者說沒一個(gè)例子可理解并運(yùn)行的。沒辦法,自己研究吧。功夫不負(fù)有心人,終有小成了,呵呵,現(xiàn)在把自己學(xué)習(xí)總結(jié)了一下,獻(xiàn)給需要的人。 DLL(動(dòng)態(tài)鏈接庫): 分WIN32 DLL和MFC DLL ActiveX:分ATL控件和MFC控件兩類(也是一個(gè)DLL) WEB:JAVASCRIPT 調(diào)用-> ActiveX調(diào)用-> DLL 完成加法運(yùn)算并返回值,在頁面上顯示。 二、開發(fā)(VS2008) 1、DLL 庫編寫: 文件-》新建-》WIN32控制臺(tái)->填寫項(xiàng)目名稱-》選擇DLL-》空項(xiàng)目-》完成。 (1)在解決方案面板中,加入一個(gè)頭文件testdll.h,內(nèi)容:
復(fù)制代碼 代碼如下: #ifndef _DLLTUT_DLL_H_ #define _DLLTUT_DLL_H_ #if defined DLL_EXPORT #define DECLDIR __declspec(dllexport) #else #define DECLDIR __declspec(dllimport) #endif //extern "C"告訴編譯器該部分可以在C/C++中使用。 extern "C" { DECLDIR int Add( int a, int b ); DECLDIR void Function( void ); } #endif(2)在解決方案面板中,加入一個(gè)實(shí)現(xiàn)文件testdll.cpp,內(nèi)容:
復(fù)制代碼 代碼如下: #include <iostream> #define DLL_EXPORT #include "testdll.h" extern "C" { // 這里主要用到 ADD 方法。 DECLDIR int Add( int a, int b ) { return( a + b ); } DECLDIR void Function( void ) { std::cout << "DLL Called!" << std::endl; } }(3)可選。新建一個(gè)WIN32控制臺(tái)類,測試這個(gè)DLL。 文件-》新建-》WIN32控制臺(tái)->填寫項(xiàng)目名稱-》選擇控制臺(tái)程序-》空項(xiàng)目-》完成。 在解決方案面板中,加入一個(gè)實(shí)現(xiàn)文件loaddll.cpp 內(nèi)容:
復(fù)制代碼 代碼如下: #include <iostream> #include <windows.h> using namespace std; typedef int (*AddFunc)(int,int); //定義指針函數(shù)、接口。 typedef void (*FunctionFunc)(); int main() { AddFunc _AddFunc; FunctionFunc _FunctionFunc; cout <<"---獲取DLL---."<< endl; // L 表示使用UNICODE 字符集,要和項(xiàng)目的字符集保持一致。 HINSTANCE hInstLibrary = LoadLibrary(L"E:\\Project\\VS\\LoadDll\\Release\\TestDll.dll"); if (hInstLibrary == NULL) { cout <<"Dll 加載【失敗】."<< endl; FreeLibrary(hInstLibrary); }else{ cout <<"Dll 加載【成功】."<< endl; } _AddFunc = (AddFunc)GetProcAddress(hInstLibrary, "Add"); _FunctionFunc = (FunctionFunc)GetProcAddress(hInstLibrary, "Function"); if ((_AddFunc == NULL) || (_FunctionFunc == NULL)) { FreeLibrary(hInstLibrary);//釋放 }else{ cout <<"---獲取DLL函數(shù)【OK】---."<< endl; } cout << _AddFunc(1, 1) << endl; // 開始調(diào)用 _FunctionFunc(); // cin.get(); // 獲得焦點(diǎn),這樣就不會(huì)程序就不會(huì)一閃而過了。 FreeLibrary(hInstLibrary);//調(diào)用完后,要釋放內(nèi)存。 return(1); }2、ActiveX 控件實(shí)現(xiàn): 這里我們選擇ATL控件實(shí)現(xiàn),而非MFC ActiveX。 文件-》新建-》ATL項(xiàng)目->填寫項(xiàng)目名稱(“FROMYANTAI”)-》選擇動(dòng)態(tài)鏈接庫(DLL)-》完成。 完成后,會(huì)在右邊“解決方案資源管理器”生成很多頭H文件和CPP實(shí)現(xiàn)文件,這些都是默認(rèn)的不要修改。 (1)、添加一個(gè)ALT簡單對象:鼠標(biāo)郵件點(diǎn)擊項(xiàng)目名稱(剛才起的名字)選擇-》添加類-》選擇ATL簡單對象。 下一步起一個(gè)名字:“ytiicrj”—》下一步:其他不變,在支持中,選擇“連接點(diǎn)”和“IE對象支持”—》完成。 下一步給“ytiicrj”添加一個(gè)方法,以便WEB頁面調(diào)用。在“類視圖”選擇“iytiicrj”(有個(gè)灰色的鑰匙圖標(biāo))鼠標(biāo)右鍵添加-》添加方法。方法起名為“GetContent”-》參數(shù)屬性選擇IN,參數(shù)類型選擇LONG 參數(shù)名 A –》添加;繼續(xù);參數(shù)屬性選擇IN,參數(shù)類型選擇LONG 參數(shù)名 B –》添加;繼續(xù);參數(shù)屬性選擇OUT和RETVAL ,參數(shù)類型選擇LONG* 參數(shù)名 out –》添加---》 點(diǎn)擊完成。 這樣就在ytiicrj.H頭文件中添加了一個(gè)(在最后一行): STDMETHOD(GetContent)(LONG a, LONG b, LONG* out); 并在ytiicrj.CPP文件中添加了一個(gè)實(shí)現(xiàn)類:
復(fù)制代碼 代碼如下: STDMETHODIMP CCaluNumCtrl::GetContent(LONG a, LONG b, LONG* out) { // TODO: 在此添加實(shí)現(xiàn)代碼 return S_OK; }(2)、在ytiicrj.H 文件中,調(diào)用DLL類庫。代碼如下: // CaluNumCtrl.h : ytiicrj 的聲明 黑體(粗體)部分是具體的實(shí)現(xiàn),其他未動(dòng)。
復(fù)制代碼 代碼如下: #pragma once #include "resource.h" // 主符號(hào) #include <windows.h> //添加 #include "AtlActiveX_i.h" #include "_ICaluNumCtrlEvents_CP.h" #if defined(_WIN32_WCE) && !defined(_CE_DCOM) && !defined(_CE_ALLOW_SINGLE_THREADED_OBJECTS_IN_MTA) #error "Windows CE 平臺(tái)(如不提供完全DCOM 支持的Windows Mobile 平臺(tái))上無法正確支持單線程COM 對象。定義_CE_ALLOW_SINGLE_THREADED_OBJECTS_IN_MTA 可強(qiáng)制ATL 支持創(chuàng)建單線程COM 對象實(shí)現(xiàn)并允許使用其單線程COM 對象實(shí)現(xiàn)。rgs 文件中的線程模型已被設(shè)置為“Free”,原因是該模型是非DCOM Windows CE 平臺(tái)支持的唯一線程模型。" #endif // ytiicrj class ATL_NO_VTABLE Cytiicrj : //增加一下一行:安全提示解除,--當(dāng)運(yùn)行瀏覽器調(diào)用時(shí),不會(huì)提示安全問題。 public IObjectSafetyImpl<Cytiicrj, INTERFACESAFE_FOR_UNTRUSTED_CALLER| INTERFACESAFE_FOR_UNTRUSTED_DATA>, public CComObjectRootEx<CComSingleThreadModel>, public CComCoClass<Cytiicrj, &CLSID_CaluNumCtrl>, public IConnectionPointContainerImpl<Cytiicrj>, public CProxy_ICaluNumCtrlEvents<Cytiicrj>, public IObjectWithSiteImpl<Cytiicrj>, public IDispatchImpl<ICaluNumCtrl, &IID_ICaluNumCtrl, &LIBID_AtlActiveXLib, 1, 0> { public: //以下三行實(shí)現(xiàn)定義。 typedef int (*AddFunc)(int,int); //類型定義,對應(yīng)DLL ADD方法。Func自定義,隨便寫。 HINSTANCE hInstLibrary; AddFunc _AddFunc; //類映射 Cytiicrj() { //開始調(diào)用DLL,進(jìn)行計(jì)算。 hInstLibrary = LoadLibrary(L"TestDll.dll");//把寫好的DLL文件放在此項(xiàng)目生成的目錄下 if (hInstLibrary == NULL) { FreeLibrary(hInstLibrary);//資源釋放 }else{ } //調(diào)用方法,返回方法句柄。 _AddFunc = (AddFunc)GetProcAddress(hInstLibrary, "Add"); } DECLARE_REGISTRY_RESOURCEID(IDR_CALUNUMCTRL) BEGIN_COM_MAP(Cytiicrj) COM_INTERFACE_ENTRY(ICaluNumCtrl) COM_INTERFACE_ENTRY(IDispatch) COM_INTERFACE_ENTRY(IConnectionPointContainer) COM_INTERFACE_ENTRY(IObjectWithSite) //增加一下一行:安全提示解除,--當(dāng)運(yùn)行瀏覽器調(diào)用時(shí),不會(huì)提示安全問題。 COM_INTERFACE_ENTRY(IObjectSafety) END_COM_MAP() BEGIN_CONNECTION_POINT_MAP(Cytiicrj) CONNECTION_POINT_ENTRY(__uuidof(_ICaluNumCtrlEvents)) END_CONNECTION_POINT_MAP() DECLARE_PROTECT_FINAL_CONSTRUCT() HRESULT FinalConstruct() { return S_OK; } void FinalRelease() { FreeLibrary(hInstLibrary); } public: STDMETHOD(GetContent)(LONG a, LONG b, LONG* out); }; OBJECT_ENTRY_AUTO(__uuidof(CaluNumCtrl), Cytiicrj)(3)、回到在ytiicrj.PP 文件中,添加實(shí)現(xiàn)代碼如下:
復(fù)制代碼 代碼如下: STDMETHODIMP CCaluNumCtrl::GetContent(LONG a, LONG b, LONG* out) { // TODO: 在此添加實(shí)現(xiàn)代碼 int sum = this->_AddFunc(static_cast<int>(a),static_cast<int>(b)); *out = static_cast<LONG>(sum); this->_AtlFinalRelease(); return S_OK; }(4)、生成DLL: 這步很簡單,選擇 Release模式,點(diǎn)擊項(xiàng)目進(jìn)行生成(會(huì)提示選擇REG32注冊,那就選擇被)。這樣就在Release目錄下生成了很多文件,我們要的就是一個(gè)DLL文件。 3、DLL和 ATL ActiveX 控件DLL 打包為CAB文件: 例如:生成test.CAB后,WEB頁面就會(huì)提示下載安裝。 (1)首先定義setup.inf文件:它描述了下載的內(nèi)容和目標(biāo)目錄還有版本號(hào)及相應(yīng)的DLL文件。這個(gè)要手動(dòng)編寫的,我的內(nèi)容如下(對應(yīng)名稱自行修改吧):
復(fù)制代碼 代碼如下: [version] ; version signature (same for both NT and Win95) do not remove signature="$CHICAGO$" AdvancedINF=2.0 [Add.Code] AtlActiveX.dll=AtlActiveX.dll TestDll.dll=TestDll.dll setup.inf=setup.inf [install.files] AtlActiveX.dll=AtlActiveX.dll TestDll.dll=TestDll.dll setup.inf=setup.inf [AtlActiveX.dll] clsid={4AE870B5-C7FB-4171-A47E-7F57AFD86F67} file-win32-x86=thiscab FileVersion=1,0,0,1 DestDir=11 RegisterServer=yes [TestDll.dll] file-win32-x86=thiscab DestDir=11 FileVersion=1,0,0,1 RegisterServer=yes [setup.inf] file=thiscab [RegisterFiles] %11%\AtlActiveX.dll ; end of INF file(2)整合資源: 將所用到的DLL全部放到一個(gè)目錄下包括setup.inf文件,然后在開始運(yùn)行:IExpress 命令去生成CAB包。 運(yùn)行后,選擇第一個(gè),下一步,選擇第三個(gè),下一步,添加文件(選擇你的DLL和INF文件),下一步,選擇一個(gè)輸出目錄并創(chuàng)建一個(gè)CAB文件名,再選擇第二個(gè)選項(xiàng),下一步,選擇第二個(gè)選項(xiàng),然后OK。這樣就生成了一個(gè)CAB文件。 (3)WEB頁面調(diào)用 ActiveX 控件 進(jìn)行加法運(yùn)算 : 寫一個(gè)test.htm網(wǎng)頁和CAB文件放在一個(gè)目錄,test.htm內(nèi)容如下:
復(fù)制代碼 代碼如下: <HTML> <HEAD> <TITLE>New Page</TITLE> <OBJECT id=CaluNumCtrl align="CENTER" WIDTH=0 HEIGHT=0 codeBase="test.CAB#version=9,0,0,1" classid="CLSID:B6D4B406-9CC4-4C80-B7A2-248BBB07F682"></OBJECT> <script language="javascript"> function doTest() { var sum = CaluNumCtrl.GetContent(1,1); alert(sum); } </script> </HEAD> <BODY> <input type="button" value="renjie" id="btnOK" οnclick="doTest();"></input> </BODY> </HTML>說明: codeBase="test.CAB#version=9,0,0,1" codeBase表示文件相對或者絕對路徑;version表示版本號(hào),如果這個(gè)號(hào)和INF文件的版本號(hào)一樣,那么第二次訪問頁面就不會(huì)下載,否則每次都下載。CLSID 是 ActiveX 項(xiàng)目生成的序號(hào),具體可以在項(xiàng)目的*.rgs 文件中找到。 好了。所有的步驟都完成了,這時(shí)你運(yùn)行test.htm,提示ActiveX控件,你選擇允許,然后就可以調(diào)用加法運(yùn)算了。 這只是一個(gè)簡單的例子,在其中的DLL中,你可以實(shí)現(xiàn)自己的應(yīng)用了。
總結(jié)
以上是生活随笔為你收集整理的DLL+ ActiveX控件+WEB页面调用例子的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: C/S框架-发布和部署WebServic
- 下一篇: DLL中导出函数的声明有两种方式