繼承情景
我們知道一個空的類,也就是其內(nèi)部沒有非靜態(tài)數(shù)據(jù)成員,沒有虛指針(包括指向虛函數(shù)表和虛基類子對象的指針),它的大小通常為1,當(dāng)然在某些對齊要求嚴格系統(tǒng)上可能是另一個數(shù)(通常是4),如果空類被繼承,那么派生類的大小會怎么樣呢?一個支持C++標(biāo)準(zhǔn)和EBO的編譯器對此會進行空基類的優(yōu)化,也就是不給空的基類子對象分配空間,換句話說,空基類子對象的地址和其派生類實例的地址是相同的。從編譯器實現(xiàn)的角度來看,需要考慮繼承時的不同情況,下圖中P表示父類,C表示子類,圓形表示空類,矩形表示非空類。單繼承EBO情況如下圖所示
EBO-1反映的是空類派生自空基類,EBO-2反映的是非空類派生自空基類,EBO-3、EBO-4反映的是在繼承鏈中,對空基類的優(yōu)化能不能傳遞到后代中。多繼承EBO如下圖所示
EBO-5反映的是空類派生自兩個空基類,EBO-6反映的是非空類派生自兩個空基類,EBO-6反映的是空類派生自一個非空基類和一個空基類,EBO-7反映的是非空類派生自一個非空基類和一個空基類。以上8種情況,不論是單繼承還是多繼承,一個完全支持EBO的編譯器就應(yīng)該能把空基類部分都優(yōu)化掉。
優(yōu)化應(yīng)用
由于空基類優(yōu)化技術(shù)節(jié)省了對象不必要的空間,提高了運行效率,因此成為某些強大技術(shù)的基石,基于類型定義類如stl中的binary_function、unary_function、iterator、iterator_traits的實現(xiàn)復(fù)用;基于策略類如內(nèi)存管理、多線程安全同步的實現(xiàn)復(fù)用。當(dāng)某個類存在空類類型的數(shù)據(jù)成員時,也可考慮借助EBO優(yōu)化對象布局,例如下
1
template<typename T1,typename T2>
2
class EBO
3

{
4
private:
5
T1 m_t1;
6
T2 m_t2;
7
};
當(dāng)T1和T2為空類時,可以改進如下
1
template<typename T1,typename T2>
2
class EBO : T1, T2
3

{
4
};
更進一步,如果T1或T2為非類類型,如基本內(nèi)建類型、函數(shù)指針等;或T1和T2類型相同時,則直接繼承它們會導(dǎo)致編譯錯誤,怎么辦呢?這時可以添加一個中間層來解決,代碼如下
1
template<typename T1,typename T2,bool isSame,bool isFirstEmpty,bool isSecondEmpty>
2
class EBO_IMPL;
3
4
template<typename T1,typename T2>
5
class EBO_IMPL<T1,T2,false,false,false>
6

{
7
T1 m_t1;
8
T2 m_t2;
9
};
10
11
template<typename T1,typename T2>
12
class EBO_IMPL<T1,T2,false,true,true> : T1,T2
13

{
14
};
15
16
template<typename T1,typename T2>
17
class EBO_IMPL<T1,T2,false,true,false> : T1
18

{
19
T2 m_t2;
20
};
21
22
template<typename T1,typename T2>
23
class EBO_IMPL<T1,T2,false,false,true> : T2
24

{
25
T1 m_t1;
26
};
27
28
template<typename T1,typename T2>
29
class EBO_IMPL<T1,T2,true,false,false>
30

{
31
T1 m_t1;
32
T2 m_t2;
33
};
34
35
template<typename T1,typename T2>
36
class EBO_IMPL<T1,T2,true,true,true> : T1
37

{
38
T2 m_t2;
39
};
40
41
template<typename T1,typename T2>
42
class EBO : EBO_IMPL<T1,T2,boost::is_same<T1,T2>::value,boost::is_empty<T1>::value,boost::is_empty<T2>::value>
43

{
44
};
為了簡便,直接使用了boost中的is_same,is_empty元函數(shù)來判斷類型的屬性,實際上boost中已經(jīng)實現(xiàn)了EBO的選擇運用工具即compressed_pair類模板,研究其源碼可發(fā)現(xiàn),該工具充分考慮到了T1和T2實際類型的各種情況,is_empty的判斷是運用sizeof來比較類型大小確定的。替換compressed_pair后,代碼如下
1
template<typename T1,typename T2>
2
class EBO: boost::compressed_pair<T1,T2>
3

{
4
};
posted on 2011-07-10 12:58
春秋十二月 閱讀(2648)
評論(0) 編輯 收藏 引用 所屬分類:
Opensrc