COM不支持實(shí)現(xiàn)繼承的原因在于這種繼承方式將使得一個(gè)對(duì)象的實(shí)現(xiàn)同另外一個(gè)對(duì)象的實(shí)現(xiàn)緊緊地關(guān)聯(lián)起來。在這種情況下,當(dāng)基類的實(shí)現(xiàn)被修改后,派生類將無(wú)法正常運(yùn)行而必須被修改。這就是為什么一些用C++編寫大型程序的專家們強(qiáng)烈建議人們基于抽象類來構(gòu)建應(yīng)用程序。
因此,為了保證以組件的修改不會(huì)影響應(yīng)用程序的正常運(yùn)行,COM并不支持實(shí)現(xiàn)繼承。我們可以用組件包容來完全模擬實(shí)現(xiàn)繼承。
包容和聚合
對(duì)一個(gè)組件加以擴(kuò)展或改造以使符合自己的使用需要,并可能會(huì)希望用此改造后的組件來代替原有的組件。
在C++中,對(duì)類的改造是用包容和繼承來實(shí)現(xiàn)的。在COM中,則可使用包容和聚合來對(duì)組件進(jìn)行改造。
包容和聚合實(shí)際上是使用一個(gè)組件使用別外一個(gè)組件的一種技術(shù)。在包容的情況下,外部組件將包含內(nèi)部組件,而在聚合的情況下,則稱外部組件聚合內(nèi)部組件。
包容的簡(jiǎn)介
包容是在接口級(jí)別完成的。外部組件包含指向內(nèi)部組件接口的指針。此時(shí)外部組件只是內(nèi)部組件的一個(gè)客戶,它將使用內(nèi)部組件的接口來實(shí)現(xiàn)它自己的接口,見圖8-1

外部組件也可以通過將調(diào)用轉(zhuǎn)發(fā)給內(nèi)部組件的方法重新實(shí)現(xiàn)內(nèi)部組件所支持的某個(gè)接口。并且外部組件還可以在內(nèi)部組件代碼的前后加一些代碼以對(duì)接口進(jìn)行改造,見圖8-2

包容的實(shí)現(xiàn)
在下面給出的例子中,組件1是一個(gè)實(shí)現(xiàn)了兩個(gè)接口IX和IY的外部組件,但它復(fù)用了組件2(組件1所包容的一個(gè)內(nèi)部組件,對(duì)IY的實(shí)現(xiàn),這一點(diǎn)同圖8-2所示完全是相同的)
包容例子
代碼下載:http://www.box.net/shared/lp661oxigu
聚合簡(jiǎn)介
聚合是包容的一個(gè)特例。外部組件將直接把內(nèi)部組件的接口指針返回給客戶。使用此種方法,外部組件將無(wú)需重新實(shí)現(xiàn)并轉(zhuǎn)發(fā)接口中的所有函數(shù)。見圖8-3。但這樣外部組件無(wú)法對(duì)接口中的函數(shù)進(jìn)行任何改造。當(dāng)外部組件將內(nèi)部組件的接口指針返回給客戶之后,客戶就可以直接同內(nèi)部組件打交道了。(但是客戶不應(yīng)該知道它是在同兩個(gè)不同的組件交互,否則將無(wú)法滿足封裝的要求。)
聚合的實(shí)現(xiàn)
假定客戶向外部組件請(qǐng)求接口IY。此時(shí)外部組件可以不實(shí)現(xiàn)IY接口,而只需將內(nèi)部組件請(qǐng)求查詢此IY接口。指針返回給客戶。客戶可以直接使用此指針來調(diào)用內(nèi)部組件所實(shí)現(xiàn)的那些IY成員。此時(shí)就IY接口而言,外部組件相當(dāng)于是被架空了;它放棄了對(duì)IY接口的控制,而將此控制交給了內(nèi)部組件。