继承 6个月前

编程语言
146
继承

一、继承方式和访问方式

注意:友元不能继承,静态成员只会在父类创造一次

父类如果用private修饰的成员,子类无法访问 protected修饰的子类可以访问,但除此之外不可访问

继承方式为private,struct为public ,一般继承方式都为public 基类的成员在子类的访问方式 = ==Min(成员在基类的访问限定符,继承方式)

二、赋值兼容规则

int main()
{
    //1.子类对象可以赋值给父类对象/指针/引用
    person p;
    student s;
    p = s;//对象赋值
    s = p;//不行
    
    //子类赋值给父类,通过切割(切片)可以进行赋值
    person* ptr = &s; //指针
    person& ref = s;  //引用
    
    //student* sptr = &p;            //父类赋值给子类,不行
    
    student* sptr = (student*)&s; //ptr指向的本来就是子类,强制转换下可以赋值    
    
    sptr = (student*)&p; //强行转换可以运行,但容易导致越界访问,访问到子类独有的类
    
    //也可以通过C++11中的类型操作符来进行试探
    sptr = dynamic_cast<student*>(ptr); //注意此时父类必须有虚函数构成多态
    //如果ptr指向父类,返回nullptr
    
    //student& sref = p;//引用赋值,不行

    return 0;
}

三、默认函数的继承

class person
{
public:
    person(const char* name = "peter")
        :_name(name)
    {
        cout << "person()" << endl;
    }

    person(const person& p)
        :_name(p._name)
    {
        cout << "person(const person& p)" << endl;
    }

    person& operator=(const person& p)
    {
        cout << "person& operator=(const person& p)" << endl;
        if (this != &p)
        {
            _name = p._name;
        }
        return *this;
    }

    ~person()
    {
        cout << "~person()" << endl;
    }

protected:
    string _name;
};


class student : public person
{
public:
    student(const char* name, int num)
        ://_name(name) 父类归父类管,所以这样初始化错误
        person(name) //直接调用父类的构造就行了
        , _num(num)
    {
        cout << "student()" << endl;
    }

    student(const student& s)
        ://_name(s._name)
        person(s) //同上
        ,_num(s._num)
    {
        cout << "student(const student& s)" << endl;
    }

    student& operator=(const student& s)
    {
        cout << "student& operator=(const student& s)" << endl;
        if (this != &s)
        {
            //operator=(s); 此时会无限递归,自己调用自己,要声明域才行
            person::operator=(s);
            _num = s._num;
        }
        return *this;
    }

    ~student() //子类的析构函数和父类的析构函数构成 隐藏(重定义) ,他们的名字会被编译器同一处理成destructor 
    {
        //person::~person(); //派生类没必要再调用父类的析构函数,析构的顺序不对,不符合栈形式
        //结束时会自动调用父类的析构函数,因为这样才能保证先析构子类,再析构父类
        //并且如果再调用会导致重复释放同一块空间
        cout << "~student()" << endl;
    }

private:
    int _num;
};

int main()
{
    student s1("jack", 18);
    student s2(s1);
    student s3("rose", 19);
    s1 = s3;
    return 0;
}

四、不能被继承的类

1.将父类全部私有,使得子类无法调用父类的析构 2.利用final关键字

class A final{public:A(){}};

class B :public A{}; //此时B就无法继承A

五、菱形继承


1.菱形继承的问题

//单继承:一个子类只有一个直接定义父类时称为单继承
class person{};
class student : public person{};
class teacher : public student{};

//多继承:一个子类有两个以上或以上直接父类时称为多继承
class student{};
class teacher{};
class assistant : public student,public teacher{};

//菱形继承
                          class person {};

class student : public person {};    class teacher : public student {};

           class assistant : public student, public teacher {};

//菱形继承的问题是:数据冗余和二义性

菱形继承时有两个问题 数据冗余:即派生类有多份相同类型的数据 二义性: 即因为有多相同类型的数据而导致对数据处理时,不知道要处理谁

class person
{
public:
    string _name;
};

class student :virtual public person //中间人都加上virtual 即可解决
{
protected:
    int _num;
};

class teacher :virtual public person //中间人都加上virtual 即可解决
{
protected:
    int _id;
};

class assistant : public student, public teacher
{
protected:
    string _course;
};

void main()
{
    assistant a;
    a._name = "peter"; //此时二义性导致不知道要赋给哪个_name,加上关键词virtual即可解决二义性和冗余性

    //显示指定访问可以解决,但是冗余性还是没有解决(两个name)
    a.student::_name = "xxx";
    a.teacher::_name = "yyy";
}

2.菱形继承virtual解决的原理

class A
{
public:
    int _a;
 //int _a[10000];
};

class B :virtual public A
{
public:
    int _b;
};

class C :virtual public A
{
public:
    int _c;
};

class D : public B, public C
{
public:
    int _d;
};

int main()
{
    D d;
    cout << sizeof(d) << endl;

    d.B::_a = 1;
    d.C::_a = 2;
    d._b = 3;
    d._c = 4;
    d._d = 5;
    d._a = 6;
    return 0;
}
//所以实际上,不到万不得已,不要把类的关系设计为菱形继承
//virtual解决了冗余性和二义性,但是会有略微的效率损失
//这个例子size变大了,但是如果父类成员size很大,那么在virtual继承后会减小很多

image

六、继承和组合

class A{};
class B : public A{};   //继承是白箱复用,父类对子类的是透明的,但是一定程度上破坏了父类的封装,继承的类是一种高耦合

class C{};
class D                    //组合是黑箱封装,C对D不透明,C保持他的封装,组合的类耦合度更低
{C c;};

//都完成了类层次的复用
//综合来看,还是组合合适

//符合 is-a  使用继承  ,eg:奔驰是车
//符合 has-a 使用组合  ,eg:车有轮子
//如果都可以,优先使用组合
image
EchoEcho官方
无论前方如何,请不要后悔与我相遇。
1377
发布数
439
关注者
2244369
累计阅读

热门教程文档

HTML
32小节
Swift
54小节
Spring Boot
24小节
Typescript
31小节
QT
33小节