多態性是C++的一個重要特征。從廣義上說,多態性是指一段程序能夠處理多種類型對象的能力;具體地講,多態性就是對不同對象發出同樣的指令時,不同對象會有不同的行為。
如果程序員充分利用C++的多態性,設計程序的運行方式會更加靈活多樣,但是會帶來一些暗藏的細節問題。這些細節的漏洞也許會通過編譯,但是在某些情況下,不可預測的結果或者背離編程者初衷的結果都會導致程序變得混亂不堪,甚至產生較大的風險。為了規避這些風險,MISRA C++推薦了一些編程規則。這些規則能夠幫助程序員更加完備或者完美地實現多態性,充分體現C++相比于傳統C語言的一些優勢。
本文主要介紹兩類在實現形式的多態性中需要注意的一些問題:一是運算符的重載,這是編譯時的多態性,即程序在編譯時就能根據重載的情況確定需要調用的函數;二是虛函數的使用,這是運行時的多態性,即在程序執行前,無法根據函數名和參數來確定調用哪個函數,必須在程序執行過程中,根據執行的具體情況來動態確定。
1 運算符的重載
運算符重載就是定義某個運算符對于某個類的具體含義。通過運算符的重載,程序員可以針對一些特定的類型使用重載的運算符含義。
規則5-2-11(強制):逗號(,),與(&&)以及或(||)運算符不允許被重載。
如果getValue和setValue的返回類型使用重載運算符&&,則這兩個函數都需要計算。
C++的內部規定是,&&和||都是在已知結果的情況下不再計算后面的值,比如0&&(a--)&& (b++)。然而重載&&運算符和||運算符導致了程序運行時要計算所有的表達式。這對于一些使用&&做判斷的運算來說,會導致一些錯誤。比如getchar()&&putchar(),在讀取文件時,如果讀到文件尾部,即得到getchar()為0時,就不需要再執行putchar()了,這樣才能正確地讀取并輸出文件。如果重載&&運算符,那么先需要計算getchar()和 putchar()的結果,再執行&&運算符的重載定義,這樣可能會導致一些不可知的錯誤。這樣的重載,會導致編譯器在處理&&和||運算符時產生混亂,所以是比較危險的。
對于逗號表達式來說,默認情況下,編譯器按照逗號表達式規定的順序計算各個表達式。但是如果重載操作逗號表達式,因為需要先檢查逗號兩邊的表達式類型,來判斷是否使用重載定義的類型,所以會導致計算順序的混亂。這樣比較危險,會產生一些不可知的錯誤。雖然在C++并沒有限制這3個運算符的重載問題,但是從這個例程和MISRA C++的規則來看,有些時候會產生一些不可預知的錯誤,所以MISRA C++不允許重載上面3個運算符。
規則5-3-3(強制):單目運算符&不允許被重載。
f1.cc和f2.cc的區別就在于f1.cc只聲明了A類,而f2.cc包含了A.h。f1.cc僅聲明A類,不會使用A類定義的重載運算,所以 f1.cc的8L運算符使用C++內部的取地址定義。f2.cc包含了頭文件A.h,因為A.h包含了A類的完整定義,所以f2.cc的&運算符就會使用用戶定義的重載操作。在同樣一個工程中,僅僅是對A類的聲明不同,就導致了在f2.cc中,&a使用用戶定義的&運算符含義,而在f1.cc中,&a使用C++內部定義的&運算符含義。
這樣差別會導致程序員在重載&運算符后,無法得知&運算符有沒有使用重載的定義。這樣做是比較危險的,可能會產生與程序員意愿不同的結果。雖然在C++中并沒有限制對單目運算符的重載操作,但是從上面的例程可以看出,MISRA C++不允許重載&運算符是很有必要的。
2 虛函數的使用
虛函數是C++中一類特殊的函數。在基類中定義一個虛函數,就說明該函數在派生類中可能有不同的實現方式。當派生類的實例調用這個虛函數時,首先會在派生類中去查看該函數有沒有被定義。如果派生類定義了這個函數,則執行派生類的函數;否則,在派生路徑上尋找最近的該函數的定義,并調用該函數。
如果從基類派生出多個派生類,那么每個派生類都可以重新定義這個虛函數。如果通過基類的指針指向派生類的對象,并訪問該虛函數,會對應地調用每個派生類的函數定義。這樣通過基類類型的指針,就可以使屬于不同派生類的對象產生不同的行為,從而實現了運行過程的多態。
關于虛函數,MISRA C++有以下幾條規則:
規則10-3-1(強制):在每一個繼承路徑上,虛函數只能有一個定義。防止按優先度調用。
例外:析構函數可以定義為虛函數,在每一個派生類上都可以有定義。
如果一個函數在同一個類中被聲明為純虛函數,但是還有定義,這樣的定義就會被忽略。
在例程的后半段是關于按優先度調用的解釋,表1顯示的是例程中每個函數的調用和定義關系。b2.f1()是按照正常的繼承關系來調用foo()函數,并且調用的是V類中foo()的定義。d.f2()和d.f1()都是按照優先度調用的。它們雖然最后都是調用了foo()函數,但是經過的繼承路徑卻不相同,而且它們最后只能調用到B1類中foo()的定義。為了防止這種情況發生,所以Misra C++規定,虛函數在一個繼承路徑上,只能有一個函數定義。
例程的前半部分描述了多個類的繼承關系,每個類都包括對幾個函數的定義和聲明。這里簡單介紹一下f1()函數,讀者可以通過表2的內容來理解其他函數。 f1在A類中是虛函數,而且有定義,在C類中有定義,所以當D類繼承C類時,D類中就不能再有定義(“√”表示可以定義,“*”表示不推薦再繼續定義)。例外是f4,雖然它在A類中有定義,但是因為它是純虛函數,所以它的定義會被忽略。
這個規則說明,如果在一個繼承路徑上有兩個函數定義,在調用函數時,有可能按照繼承的優先度調用函數。這樣就會導致函數調用的混亂,可能會調不到程序員希望的函數。這是在實現多態時需要特別注意的地方。關于繼承路徑上的函數定義,C++并沒有明確限制。
從上面的例程可以看出,如果沒有這樣的限制,就會產生一些混亂,雖然程序能夠正常運行,但是不一定能夠按照程序員所設計的方式運行。這樣的運行方式會出現很多漏洞,所以MISRA C++強制規定在每一個繼承路徑上,虛函數只能有一個定義。
規則10-3-2(強制):每一個重載的虛函數應該用關鍵字virtual來聲明。
這樣做不需要檢查基類,就可以確定函數是否為虛函數。MISRA C++推出這樣的規則是為了使C++程序更加完善。
規則10-3-3(強制):只有被聲明為純虛函數的虛函數,才能被純虛函數重載。
foo函數在A類中定義為純虛函數,在B類中被重載為普通虛函數。而C類使用純虛函數重載foo函數。這樣做是不行的。
B類中foo函數重載A類的foo函數時,是用有定義的虛函數重載純虛函數,這樣做是可以的。
C類中的foo函數重載B類的foo函數時,是用純虛函數重載一個非純虛函數,這樣是不行的。在C類中,foo被定義為純虛函數,在C類的對象調用foo 函數時無法調用到B類中的定義。這樣的重載導致B類中對foo函數的定義丟失。
所以MISRA C++不允許使用純虛函數重載非純虛函數,這樣做的目的也是為了使C++程序更加安全。
3 小結
正確并完備地實現C++的多態性,能夠充分發揮C++的優勢,并且提高程序的可讀性和可維護性。如果使用不當,會導致一些想象不到的程序漏洞。MISRA C++針對使用多態性可能產生的一些漏洞,提出了規避的方法與建議。本文列出了其中幾條比較關鍵和實用的規則。關于多態性的其他規則,讀者可以查看。 MISRA C++(2008),以避免不正確使用多態性所導致的一些程序漏洞。
責任編輯:gt
-
C++
+關注
關注
22文章
2114瀏覽量
73771 -
編譯器
+關注
關注
1文章
1640瀏覽量
49222
發布評論請先 登錄
相關推薦
評論