[OOD-More C++ Idioms] 內(nèi)部類 (Inner Class)
內(nèi)部類 (Inner Class)
目的
不用通過多重繼承就可以實現(xiàn)多套接口,同時可以自然地向上轉(zhuǎn)換(Up-casting)。
在單個抽象下提供相同接口的多個實現(xiàn)。
別名
動機
兩個獨立類庫通過不同的接口提供的虛函數(shù)簽名可能沖突,如果這時需要同時實現(xiàn)這兩個函數(shù)就會出現(xiàn)問題。示例如下:
class Base1 /// 來自月球
{
public:
virtual int open (int) = 0;
/* virtual */ ~Base1() {} // 不允許多態(tài)的析構(gòu)函數(shù)
};
class Base2 /// 來自木星
{
public:
virtual int open (int) = 0;
/* virtual */ ~Base2() {} // 不允許多態(tài)的析構(gòu)函數(shù)
};
class Derived : public Base1, public Base2
{
public:
virtual int open (int i)
{
// Wow! 到底來自哪里?
return 0;
}
/* virtual */ ~Derived () {}
};
內(nèi)部類慣用法就是用來解決這個問題。
解決方案及示例
仍然是上面的例子,兩個基類不用修改,改用如下方式實現(xiàn)子類:
#include <iostream>
class Base1 /// 來自月球
{
public:
virtual int open (int) = 0;
/* virtual */ ~Base1() {} // 不允許多態(tài)的析構(gòu)函數(shù)
};
class Base2 /// 來自木星
{
public:
virtual int open (int) = 0;
/* virtual */ ~Base2() {} // 不允許多態(tài)的析構(gòu)函數(shù)
};
class Derived // 注意沒有繼承
{
class Base1_Impl;
friend class Base1_Impl; // 注意聲明友元
class Base1_Impl : public Base1 // 注意是公共繼承
{
public:
Base1_Impl(Derived* p) : parent_(p) {}
int open() override { return parent_->base1_open(); }
private:
Derived* parent_;
} base1_obj; // 注意成員變量.
class Base2_Impl;
friend class Base2_Impl; // 注意聲明友元
class Base2_Impl : public Base2 // 公共繼承
{
public:
Base2_Impl(Derived* p) : parent_(p) {}
int open() override { return parent_->base2_open(); }
private:
Derived* parent_;
} base2_obj; // 成員變量
int base1_open() { return 111; } /// 實現(xiàn)
int base2_open() { return 222; } /// 實現(xiàn)
public:
Derived() : base1_obj(this), base2_obj(this) {}
operator Base1&() { return base1_obj; } /// 轉(zhuǎn)到Base1&
operator Base2&() { return base2_obj; } /// 轉(zhuǎn)到Base2&
}; /// class Derived
int base1_open(Base1& b1) { return b1.open(); }
int base2_open(Base2& b2) { return b2.open(); }
int main(void) {
Derived d;
std::cout << base1_open(d) << std::endl; // Like upcasting in inheritance.
std::cout << base2_open(d) << std::endl; // Like upcasting in inheritance.
}
附個類圖便于理解:
inner_class
這里的類Derived并不是子類,而是通過內(nèi)部的兩個嵌套類實現(xiàn)不同的接口,再橋接回到自己定義的兩個實現(xiàn)的函數(shù): base1_open及base2_open。兩個嵌套類不會共享繼隨關(guān)系,通過Derived類提供的兩個轉(zhuǎn)換操作符可以實現(xiàn)Derived轉(zhuǎn)換到任意的基類。另外兩個內(nèi)部類對象也免去了額外的生命周期管理,它們的生命周期與Derived對象一致。
已知的應(yīng)用
譯注:
Inner Class的概念來自于Java, 其本特征是嵌套類通過友元的方式可以使用外部類的私有成員變量和成員函數(shù),從而支持更強的交互。而且通常這個內(nèi)部類需要是私有的。
以Chromium網(wǎng)絡(luò)模塊的Http Cache為例:
sample
這是一個簡單的例子,并沒有多重繼承。更多的是強調(diào)了封裝和信息隱藏(HttpCache::Transaction是HttpCache內(nèi)私有的類)的OO特性。
相關(guān)的慣用法
Interface Class
Capability Query
參考
Thinking in C++ Vol 2 - Practical Programming — by Bruce Eckel.