C++私有继承(一)
文章转自微信公众号:Coder梁(ID:Coder_LT)
1.私有继承介绍
我们可以将一个类作为另外一个类的成员,这样可以描述has-a
的关系,即包含关系。
例如我们要创建一个Student
类,它当中要包含一个string类型的name,和valarray类型的scores。这里的valarray是C++中的一个模板类,它可以理解成一个泛型的数组,有些类似于vector和array,但提供的功能更多。比如拥有min, size, max, sum等方法。
除了使用成员变量来描述has-a关系之外,我们还可以使用私有继承。
在私有继承当中,基类的公有成员和保护成员都会成为派生类的私有成员。这意味着基类的方法都会被private关键字描述,我们可以在派生类中使用它,但类对象无法直接调用。
2.语法
首先我们来看看私有继承的语法,其实和共有继承类似,只不过将public
关键字替换为private
,另外由于继承关系默认为private
,所以也可以不填。
using namespace std; class Student : private string, private valarray<double> { ... };
这里的private
都可以省略。
其次是构造函数中成员列表初始化的处理,如果是常规的成员定义方式,即:
class Student { private: string name; valarray<double> scores; };
我们的构造函数签名可以这么写:
Student(const string& n, const double *pd, int n): name(n), scores(pd, n) {}
但我们这个例子当中用的是私有继承,存储的值都在基类当中,所以我们只能通过类名来进行初始化:
Student(const string& n, const double *pd, int n): string(n), valarray<double>(pd, n) {}
为了书写方便,我们可以定义一个类型转换,将valarray<double> 替换为ArrayDb,于是上述的代码可以写成这样:
Student(const string& n, const double *pd, int n): string(n), ArrayDb(pd, n) {}
Student(const string& n, const double *pd, int n): string(n), ArrayDb(pd, n) {}
3.类声明代码
最后, 我们来看下完整的类声明的代码:
using namespace std; class Student: string, valarray<double> { private: typedef valarray<double> ArrayDb; ostream & arr_out(ostream &os) const; public: Student(): string("null"), ArrayDb() {} explicit Student(const string & s): string(s), ArrayDb() {} explicit Student(int n): string("null"), ArrayDb(n) {} Student(const string& s, int n): string(s), ArrayDb(n) {} Student(const string& s, const ArrayDb& a): string(s), ArrayDb(a) {} Student(const char* str, const double *pd, int n): string(str), ArrayDb(pd, n) {} ~Student() {} double Average() const; double & operator[](int i); double operator[](int i) const; const string& Name() const; friend istream & operator>>(istream &is, Student &stu); friend istream & getline(istream &is, Student& stu); friend ostream & operator<<(ostream &os, const Student &stu); };