C++ COM編程之QueryInterface函數(shù)(二)
前言
在COM編程——認識組件中也總結(jié)了,COM是一個說明如何建立可動態(tài)互變組件的規(guī)范,它提供了為保證能夠互操作,客戶和組件應(yīng)遵循的一些標準。而在實現(xiàn)和使用QueryInterface時,就需要去遵守一些規(guī)則,只有遵守了這些規(guī)則,才能是一個正確的COM組件;只有了解了這些規(guī)則,才能會真正的了解COM開發(fā)。
QueryInterface的實現(xiàn)規(guī)則
實現(xiàn)QueryInterface需要遵從以下五條規(guī)則:
1.QueryInterface總是返回同一IUnknown指針
組件的實例只有一個IUnknown接口。因為當查詢組件實例的IUnknown接口時,不論通過哪個接口,所得到的均將是同一指針值。為確定兩個接口是否指向同一個組件,可以通過這兩個接口查詢IUnknown接口,然后將返回值進行比較。
這條規(guī)則是非常重要的,如果QueryInterface的實現(xiàn)不遵循這條規(guī)則的話,則將沒法決定兩個接口是否指向同一組件;
2.如果客戶曾經(jīng)獲取過某個接口,那么它將總能獲得該接口
這條規(guī)則限定了對于一個組件實例,它的QueryInterface的不變性;你可以想象,如果組件實例的接口集不是固定的,客戶將無法通過編程的方法來決定一個組件到底具有一些什么樣的功能;客戶就會對你的COM組件失去耐心,你的COM組件都沒有人去使用了,這還有什么意義。
3.客戶可以再次獲取已經(jīng)擁有的接口
如果客戶擁有一個IX接口,則可以通過它來再次查詢IX接口指針,并且一定可以成功的。通過自己查詢自己,聽起來多少有點奇怪,但是這是必須可以的。
4.客戶可以從任何接口返回到起始接口
如果客戶擁有一個IX接口指針,并成功地使用它來查詢了一個IY接口,那么它將可以使用這個IY接口來查詢一個IX接口,這條規(guī)則在實際的項目開發(fā)時很有用。
5.如果能夠從某接口獲取某特定接口,則從任意接口都將能獲取此接口
如果能夠從某個組件獲取某特定接口,那么客戶將可以通過此組件所支持的任意接口獲取此接口。例如:如果可以通過接口IX得到接口IY,通過IY可以得到IZ,那么通過IX也將可以得到IZ。這條規(guī)則使得QueryInterface是可用的。
綜上所有規(guī)則,其內(nèi)在的重點在于不管組件實現(xiàn)了多少個接口,組件都只實現(xiàn)了一個QueryInterface,所以,在所有的接口的vtbl中,對應(yīng)的QueryInterface都是組件實現(xiàn)的QueryInterface的地址,所有接口指針調(diào)用QueryInterface進行查詢時,都是調(diào)用的同一個QueryInterface,所以,這就滿足了上述的規(guī)則。大家在閱讀上面的這些規(guī)則時,難免會有些無所謂的感覺,覺得都是文字,很枯燥,我開始的時候也是這樣的;就是因為如此,在實際的開發(fā)中,吃過不少的苦頭,所以,今天又在這里把這些規(guī)則重新的整理一遍,希望大家不要在實際的項目中栽了跟頭再回來尋找原因,何不防范于未然呢?
添加新的接口
以前的博文也總結(jié)過了,COM中接口是不會發(fā)生變化的。當組件發(fā)布一個接口并被某個客戶使用之后,此接口將決不會發(fā)生任何變化,而將永遠保持不變。這里說的不變,具體是什么意思呢?由于每一個接口都有一個唯一對應(yīng)的接口標識符IID。一般情況下,我們不會改變接口,而可以建立一個新接口并為之指定一個新的IID。當QueryInterface接收到對老的IID的查詢時,它將返回老的接口;而當它收到對新的IID的查詢時,它將返回新的接口。對于QueryInterface而言,一個IID就是一個接口。
所以同某個IID相應(yīng)的接口將絕不會發(fā)生變化。新接口可以繼承老的接口,它也可以同老接口完全不同。由于老的接口仍然保持不變,已有客戶的運行將不會受到任何影響。而新客戶則可以自行決定是使用老接口還是新接口,因它可以自由決定到底是查詢哪個接口。
新接口命名
雖然每個人的命名規(guī)則,每個公司命名規(guī)則都是要求不同的,但是對于COM接口的命名大體上都是一致的,例如:原來的接口名為IX,則新的接口名為IX2,而不是IXEx之類的。我經(jīng)歷了這么多的項目,寫過、也調(diào)用過很多的COM組件,基本都是遵循的這個規(guī)則,即在老名稱的后面加上一個數(shù)字。
總結(jié)
這篇文章總結(jié)的是理論,讓那些不喜歡理論的人會有點失望。但是,道理就是那樣的,沒有理論作為基礎(chǔ)的實踐,都是亂搞。做什么事情,都要有一定的理論基礎(chǔ),所以,我通過了兩篇博文,對QueryInterface進行詳細的總結(jié)。希望對大家有一定的幫助,最后,也希望大家提出你的想法和我分享。我堅信,交流是一種非常給力的學習方法。
相關(guān)文章
c++調(diào)用python實現(xiàn)圖片ocr識別
所謂c++調(diào)用python,實際上就是在c++中把整個python當作一個第三方庫引入,然后使用特定的接口來調(diào)用python的函數(shù)或者直接執(zhí)行python腳本,本文介紹的是調(diào)用python實現(xiàn)圖片ocr識別,感興趣的可以了解下2023-09-09在輸入輸出字符串時scanf(),printf()和gets(),puts()的區(qū)別淺談
在輸入輸出字符串時scanf(),printf()和gets(),puts()的區(qū)別淺談,需要的朋友可以參考一下2013-02-02深入二叉樹兩個結(jié)點的最低共同父結(jié)點的詳解
本篇文章是對二叉樹兩個結(jié)點的最低共同父結(jié)點進行了詳細的分析介紹,需要的朋友參考下2013-05-05C語言實現(xiàn)ATM自動取款機系統(tǒng)的示例代碼
ATM自動取款機系統(tǒng)是銀行業(yè)務(wù)流程中十分重要且必備的環(huán)節(jié)之一,在銀行業(yè)務(wù)流程中起著承上啟下的作用。本文將用C語言實現(xiàn)一個簡單的ATM自動取款機系統(tǒng),需要的可以參考一下2022-08-08visual studio code 編譯運行html css js文件的教程
這篇文章主要介紹了visual studio code 如何編譯運行html css js文件,本文通過圖文實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-03-03