作者 | 梁唐
大家好,我是梁唐。
这是EasyC++系列的第83篇,来聊聊私有继承。
我们可以将一个类作为另外一个类的成员,这样可以描述has-a的关系,即包含关系。
例如我们要创建一个Student
类,它当中要包含一个string
类型的name,和valarray
类型的scores。这里的valarray
是C++中的一个模板类,它可以理解成一个泛型的数组,有些类似于vector
和array
,但提供的功能更多。比如拥有min, size, max, sum
等方法。
除了使用成员变量来描述has-a关系之外,我们还可以使用私有继承。
在私有继承当中,基类的公有成员和保护成员都会成为派生类的私有成员。这意味着基类的方法都会被private
关键字描述,我们可以在派生类中使用它,但类对象无法直接调用。
首先我们来看看私有继承的语法,其实和共有继承类似,只不过将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) {}
最后, 我们来看下完整的类声明的代码:
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);
};
这只是类声明的代码,当中有很多的细节,由于篇幅限制,我们在之后的文章当中再详细讨论。