基于COM的輕量級組件技術(shù)的簡單實(shí)現(xiàn)
0 引言
在軟件開發(fā)中,組件是一些小的二進(jìn)制可執(zhí)行程序,它們可以給應(yīng)用程序、操作系統(tǒng)以及其他組件提供服務(wù)。實(shí)際應(yīng)用中主要采用COM技術(shù)開發(fā)軟件組件。這是由Microsoft提出的一種組件標(biāo)準(zhǔn),它定義了組件程序之間進(jìn)行交互的標(biāo)準(zhǔn)。標(biāo)準(zhǔn)的COM技術(shù)主要用于Microsoft Windows平臺,是windows操作系統(tǒng)和各種應(yīng)用軟件的結(jié)構(gòu)基礎(chǔ),其突出優(yōu)點(diǎn)是軟件之間直接通過二進(jìn)制接口進(jìn)行通訊。
開發(fā)自定義的COM組件就如同開發(fā)動態(tài)的、面向?qū)ο蟮腁PI。多個COM對象可以鏈接起來形成應(yīng)用程序或組件系統(tǒng)。組件可以在運(yùn)行時刻,并在不被重新鏈接或編譯應(yīng)用程序的情況下被卸下或替換掉。
要實(shí)現(xiàn)基于COM標(biāo)準(zhǔn)的跨平臺輕量級組件,需要滿足三個條件:
一是源代碼級跨平臺,可以在LINUX和WINDOWS上使用;
二是像容器一樣的管理組件DLL/SO;
三是功能組件化。
在工控系統(tǒng)組態(tài)過程中,通常存在著人際界面軟件開發(fā)周期長,維護(hù)困難,升級不便等缺點(diǎn),為了增強(qiáng)組態(tài)軟件的行業(yè)競爭力,希望從軟件開發(fā)模式上改變當(dāng)前系統(tǒng)軟件的缺陷,同時為了適應(yīng)系統(tǒng)軟件跨平臺的需要,根據(jù)COM標(biāo)準(zhǔn)組件技術(shù),并結(jié)合QT、以及組態(tài)人機(jī)界面軟件自身的特點(diǎn),可以實(shí)現(xiàn)輕量級組件技術(shù)。本文所述的輕量級組件技術(shù)就是以COM標(biāo)準(zhǔn)為基礎(chǔ)的,并根據(jù)實(shí)際的應(yīng)用環(huán)境做了某些改變。
1 輕量級組件各部分的協(xié)作關(guān)系
跨平臺輕量級組件技術(shù)可以COM標(biāo)準(zhǔn)為基礎(chǔ)來實(shí)現(xiàn),可先定制Com庫,并確定編寫組件的規(guī)范。
輕量級組件技術(shù)使用的是分層結(jié)構(gòu),該結(jié)構(gòu)通??煞譃榭蛻舳?、Com庫、組件端三個層。圖1所示是其輕量級組件的結(jié)構(gòu)圖。其中,組件端用于提供功能和接口;COM庫負(fù)責(zé)組件和客戶端的通信,客戶端則通過Com庫創(chuàng)建組件,并調(diào)用組件接口。
它們?nèi)糠忠话阃ㄟ^接口進(jìn)行通信,其具體過程如圖2所示。
在實(shí)際應(yīng)用中,Com庫一般可由圖3所示兩部分構(gòu)成。實(shí)際上,圖3中的Comku是一個動態(tài)庫,負(fù)責(zé)提供最核心的功能:ExtensionSystem靜態(tài)庫是對Comku的封裝,為了使用方便,它同時提供了生成插件接口的宏以及CPluginManager類中的接口函數(shù)。下文中如不特別指明,Com庫均表示單一的動態(tài)庫。
整個組態(tài)的人機(jī)界面軟件是在QT中實(shí)現(xiàn)的,因此組件的核心庫也需要使用QT來實(shí)現(xiàn)。QT是一個多平臺的C++圖形用戶界面應(yīng)用程序框架。它可以提供給應(yīng)用程序開發(fā)者建立藝術(shù)級的圖形用戶界面所需的所有功能。QT是完全面向?qū)ο蟮?、很容易擴(kuò)展、且可支持源代碼的級跨平臺。直接在QT環(huán)境中實(shí)現(xiàn)輕量級組件技術(shù),以達(dá)到跨平臺的目的。實(shí)際上,組件庫本身并不依賴于QT環(huán)境。
2 Com庫的原理及功能
對于使用了輕量級組件技術(shù)的應(yīng)用程序,Com庫主要用來管理插件DLL、創(chuàng)建和銷毀組件指針,它像容器一樣,可以保存當(dāng)前應(yīng)用程序中用到的插件信息和接口信息,并在用戶需要的時候返回正確的插件接口指針。[!--empirenews.page--]
2.1 管理組件的功能
Com庫中可維護(hù)兩個鏈表,分別用于存放插件DLL的信息和接口信息。
每次使用Com庫加載的DLL信息都將記錄到插件鏈表中。在為用戶創(chuàng)建接口前,還需要查詢插件鏈表,以判斷該插件是否已經(jīng)被加載,如已加載,便可直接使用,不需要重新加載,這樣可以節(jié)省系統(tǒng)開銷。由于在應(yīng)用中加載的插件數(shù)目有限,因此,查詢鏈表花費(fèi)的時間代價可以忽略,圖4所示為存放插件信息的結(jié)構(gòu)。其中CLASS_TABLE結(jié)構(gòu)存放插件DLL的信息。接口鏈表可用來保存成功返回給用戶的插件接口。用戶申請接口時,Com庫先在接口鏈表中查找該接口,如果已經(jīng)存在,則直接返回,這樣可以節(jié)省創(chuàng)建接口的時間,也可以有效地控制接口對象的數(shù)目。
圖5所示是CIMPL結(jié)構(gòu)用于存放插件接口的信息。
釋放這兩個鏈表的過程就是釋放組件資源的過程。在COM標(biāo)準(zhǔn)中,資源管理是通過一個由接口自己負(fù)責(zé)的資源計數(shù)器完成的。在輕量級組件技術(shù)中,根據(jù)應(yīng)用需要,每個接口只需要一個接口指針,而不需要資源計數(shù)器,釋放資源的任務(wù)由Com庫完成。當(dāng)應(yīng)用程序關(guān)閉時,可將兩個資源鏈表中的接口對象刪除,并將組件DLL/SO卸載。
2.2 創(chuàng)建插件接口指針
Com庫的核心功能就是創(chuàng)建組件接口指針。
創(chuàng)建組件接口指針的函數(shù)為CoCreateInstance,它具有圖6所示的三個參數(shù)。其創(chuàng)建過程如圖7所示。
2.3 ExtensionSystem靜態(tài)庫
ExtensionSystem靜態(tài)庫中已加載了Com庫,并將Com庫的創(chuàng)建接口指針函數(shù)封裝成接口。這樣,客戶端在使用時,就不需要關(guān)注Com庫的具體位置,而可以直接調(diào)用ExtensionSystem的接口來獲取插件接口指針。[!--empirenews.page--]
2.4 配置文件
配置文件主要用來標(biāo)識當(dāng)前程序所需要加載的組件,以及組件的位置。配置文件可分為兩部分:
第一部分用于標(biāo)識加載的組件和加載順序;第二部分用于標(biāo)識組件的位置。圖8所示是配置文件的信息示意圖。
DLL/SO組件編寫規(guī)則
輕量級組件技術(shù)要求組件DLL/SO應(yīng)按照約定的格式編寫,只有組件庫加載后,才能為客戶端創(chuàng)建組件接口指針。
一個類似于IUnknown的基接口,通常是所有接口都需要繼承的,其代碼如下:
Release函數(shù)可用來釋放組件接口對象。
每個DU/SO中都需要有一個類實(shí)現(xiàn)IPlugin接口,該類稱為組件類,Init函數(shù)在ExtensionSysten中被調(diào)用時,可以在其中完成組件的初始化工作,其具體代碼如下:
每個組件DLL/SO都需要導(dǎo)出一個創(chuàng)建接口的函數(shù),形式如下:
4 結(jié)束語
QT本身提供有插件機(jī)制,但是,加載方式對用戶是不可見的,因此,出現(xiàn)問題后,往往不利于調(diào)試。而本文所述的組件編寫方式和加載方式使用約定的形式,這種形式并不局限于QT中,而是在支持標(biāo)準(zhǔn)c++的編譯環(huán)境中都可以使用,因而是一種通用的輕量級組件技術(shù)。