C++ | 继承(基类,父类,超类),(派生类,子类)

这篇具有很好参考价值的文章主要介绍了C++ | 继承(基类,父类,超类),(派生类,子类)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

文章参考:https://blog.csdn.net/war1111886/article/details/8609957

一 .继承中的访问权限关系
1.基类,父类,超类是指被继承的类,派生类,子类是指继承于基类的类. 

2.在C++中使用: 冒号表示继承,如class A : public B;表示派生类A从基类B继承而来

3.派生类包含基类的所有成员,而且还包括自已特有的成员,派生类和派生类对象访问基类中的成员就像访问自已的成员一样,可以直接使用,不需加任何操作符,但派生类仍然无法访问基类中的私有成员.

4.在C++中派生类可以同时从多个基类继承,Java 不充许这种多重继承,当继承多个基类时,使用逗号将基类隔开.

5.基类访问控制符

     class A : public B        基类以公有方式被继承,

     class A:private B        基类以私有方式被继承,

     class A:protected B    基类以受保护方式被继承,如果没有访问控制符则默认为私有继承。

6. protected 受保护的访问权限:使用protected 保护权限表明这个成员是私有的,但在派生类中可以访问基类中的受保护成员。派生类的对象就不能访问受保护的成员了。

7. 如果基类以public公有方式被继承,则基类的所有公有成员都会成为派生类的公有成员.受保护的基类成员成为派生类的受保护成员
      如果基类以private私有被继承,则基类的所有公有成员都会成为派生类的私有成员.基类的受保护成员成为派生类的私有成员. 

      如果基类以protected 受保护方式被继承,那么基类的所有公有和受保护成员都会变成派生类的受保护成员.

     不管基类以何种方式被继承,基类的私有成员,仍然保有其私有性,被派生的子类不能访问基类的私有成员.

例:继承中的访问权限关系
 

class A {
        int a;
    protected:
        int b;
    public:
        int c;
 
        A() { a = b = c = 1; }
    };
 
     //类B以公有方式从基类A继承
    class B : public A {
    public:
        int d;
 
        B() {//a=2;  //错误,不能访问基类中的私有成员
            b = 2;   //正确,可以在类中访问基类中的受保护成员,但类的对象不能访问,基类中的受保护成员b在类B中仍然是受保护成员
            c = d = 2;
            }
    }; //基类中的公有成员c在类B中仍然是公有成员
 
    //类C以受保护和私有方式从基类A继承。
    class C : protected A {
    public:
        int e;
 
        C() {//a=3;  //错误,不能访问基类中的私有成员
            b = c = e = 3;
        }
    };//这里基类受保护成员b和公有成员c都成为类C中的受保护成员。
 
    class D : private A {
        public:
        D()
        { b = c = 4; }
    };//基类中的公有和受保护成员都成为了类D中的私有成员。
 
  //验证受保护和私有方式继承的访问权限。
    class C1 : public C {
    public:
        C1() { b = c = e = 4; }
    };//正确;类A中的成员b和c在类C中是以受保护方式被继承的,b和c都成为了类C中的受保护成员。
 
    class D1 : public D {
    public:
        D1() {//b=5; //错误,在A中受保护的成员b在类D中是以私有方式继承的,这样b就成为了类D中的私有成员,所以无法访问。
              //c=5; //错误,在A中公有的成员c在类D中是以私有方式继承的,这样c就成为了类D中的私有成员,所以无法访问。
             }
    };
 
        int main()
        {
            A m1;
            B m2;
            C m3;
            D m4;
            //cout<<m1.b<<m2.b<<m3.b<<m4.b<<endl;   //错误;不能用类的对象访问受保护的成员,只有在类中才能访问。
            cout << m1.c <<endl;
            cout << m2.c <<endl;
            //cout<<m3.c<<endl;    //错误,类C是以受保护的方式从A继承的,基类中的变量c在类C中就是受保护的,所以类的对象不能访问
            //cout<<m4.c<<endl;    //错误,类C是以私有的方式从A继承的,基类中的变量c在类C中就是私有的,所以类的对象不能访
        }

二.覆盖和隐藏基类成员变量或成员函数
1. 基类的成员变量或函数被覆盖:

如果派生类覆盖了基类中的成员函数或成员变量,则当派生类的对象调用该函数或变量时是调用的派生类中的版本,当用基类对象调用该函数或变量时是调用的基类中的版本。

2. 隐藏基类成员函数的情况:

如果在派生类中定义了一个与基类同名的函数,不管这个函数的参数列表是不是与基类中的函数相同,则这个同名的函数就会把基类中的所有这个同名的函数的所有重载版本都隐藏了,这时并不是在派生类中重载基类的同名成员函数,而是隐藏,比如类A中有函数 f (int i , int j)和 f (int i)两个版本,当在从A派生出的类B中定义了基类的 f() 函数版本时,这时基类中的 f (int i)和f (int i , int j)就被隐藏了,也就是说由类B创建的对象比如为m,不能直接访问类A中的f(int i)版本,即使用语句m.f(2)时会发生错误。

3. 怎样使用派生类的对象访问基类中被派生类覆盖或隐藏了的函数或变量:

      3.1. 方法1   使用作用域运算符:: ,在使用对象调用基类中的函数或变量时使用作用域运算符即语句m.A::f(2),这时就能访问基类中的函数或变量版本。注意,访问基类中被派生类覆盖了的成员变量只能用这种方法
      3.2.方法2   使用using:: , 该方法只适用于被隐藏或覆盖的基类函数,在派生类的类定义中使用语句using 把基类的字包含进来,比如using A::f;就是将基类中的函数f()的所有重载版本包含进来,重载版本被包含到子类之后,这些重载的函数版本就相当于是子类的一部分,这时就可以用派生类的对象直接调用被派生类隐藏了的基类版本,比如m.f(2),但是使用这种语句还是没法调用基类在派生类中被覆盖了的基类的函数,比如m.f()调用的是派生类中定义的函数f,要调用被覆盖的基类中的版本要使用语句m.A::f()才行。

4. 在派生类的函数中调用基类中的成员变量和函数的方法:

就是在函数中使用的被派生类覆盖的基类成员变量或函数前用作域解析符加上基类的类名,即A::f()就是在派生类的函数中调用基类中被派生类覆盖了的函数f()的方法。

5. 派生类以私有方式被继承时改变基类中的公有成员为公有的方法:

      5.1.使用:: 作用域运算符,不提倡用这种方法,在派生类的public 后面用作用域运算符把基类的公有成员包含进来,这样基类的成员就会成为派生类中的公有成员了,注意如果是函数的话后面不能加括号,如A::f;如果f是函数的话不能有括号。
      5.2.使用using语句,现在一般用这种方法,也是在派生类的public使用using把基类成员包函进来,如using A::f。

例:隐藏或覆盖基类中的成员,使用::作用域运算符访问
————————————————
版权声明:本文为CSDN博主「Sherlock_Homles」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/Sherlock_Homles/article/details/82927515

class A {
    int a;
protected:
    int b;
public:
    int c, d;
 
    void f(int i) {
        cout << "class A" << "\n";
    }
 
    A() { a = b = c = d = 1; }
};
 
class B : public A {
public:
    int d;
     //覆盖基类中的成员变量d。
    B() {
        b = c = d = 2;            //这里是给子类B中的成员变量d赋值,而不是基类中的d
        A::d = 3;
    } //给基类中被覆盖的成员d赋值,注意在类中访问的被覆盖成员的方式。
    void f() {
        cout << "class B" << "\n";  //在子类中重定义基类中的同名函数,虽然参数列表不一样,但同样会隐藏基类中的同名函数
        A::f(1);    //在函数中调用基类中被隐藏了的同名函数的方法,使用作用域解析运算符。
      //f(1);       //错误,因为基类中的函数被子类中的同名函数隐藏了,在这里子类不知道有一个带参数的函数f。
    }
};
        int main() {
            B m;
            cout << m.d << "\n";      //输出子类中的成员变量d的值,注意派生类中覆盖了基类成员d.
            cout << m.A::d << "\n";   //输出基类中的成员变量d的值,注意这是使用对象访问被覆盖的基类成员的方式
            m.f();                    //调用子类中的不带参数的函数f。
         // m.f(2);                   //错误,因为基类中的带一个参数的函数f被子类中的同名函数隐藏掉了,不能这样访问,须用作用域解析运算符来访问。
            m.A::f(1);
        } //使用子类对象访问基类中被隐藏的函数的方法。

例:使用using 语句以便访问基类中被隐藏的函数

class A {
    int a;
protected:
    int b;
public:
    int c, d;
 
    void f() { cout << "Amoren" << "\n"; }
 
    void f(int i) { cout << "class A" << "\n"; }
 
    A() { a = b = c = d = 1; }
};
 
class B : public A {
public:
    int d;
     //覆盖基类中的成员变量d。
    B() {
        b = c = d = 2;  //这里是给类B中的成员变量d赋值,而不是基类中的d
        A::d = 3;
    } //给基类中被覆盖的成员d赋值,注意在类中访问的被覆盖成员的方式。
    using A::f;        //使用语句using把类A中的函数f包含进来,以便以后可以直接访问基类被隐藏了的函数,注意函数f没有括号
 
    void f() {
        cout << "class B" << "\n";  //在子类中覆盖基类中的同名函数,注意这里是覆盖,同时会隐藏基类中的其他同名重载函数
        f(1);       //正确,因为使用了using语句,所以可以在类中直接使用基类中f函数的重载版本。
        A::f(2);    //正确,虽然使用了using语句,但同样可以按这种方法访问基类中的函数。
        A ma;
        ma.f();     //正确,在子类中创建的基类对象,可以直接用对象名调用基类中被子类覆盖或隐藏了的函数,因为这时不会出现二义性。
        ma.f(1);    //正确,在子类中创建的基类对象,可以直接用对象名调用基类中被子类覆盖或隐藏了的函数,因为这时不会出现二义性。
    }
    
    void g() {
        cout << "this g" << "\n"; 
        f();        //正确,但该语句访问的是子类中的不带参数函数f,虽然在类中使用了using语句,但直接调用被子类覆盖了的基类函数时不能使用这种方法
        A::f();     //正确,调用被子类覆盖了的基类中的函数f,注意,虽然使用了using但要访问被子类覆盖了的函数,只能这样访问。
    }
};
int main() {
    B m;
    m.f();       //调用子类中的不带参数的函数,这里不会调用基类中的不带参数的被覆盖的函数f。
    m.A::f();    //调用基类中被子类覆盖了的函数f,虽然子类使用了using语句,但要访问基类中被覆盖的方法只能像这样使用。
    m.f(1);      //调用基类重载的f函数,注意这里可以不用::运算符,因为在子类中使用了using,只要子类没有覆盖基类中的方法,都可以这样直接调用。
    m.A::f(2);   //当然,使用了using后,也可以使用这种方法
} 

例:派生类以私有方式被继承时改变基类中的公有成员为公有的方法

class A {
public:
    int a, b;
    void f() { cout << "f" << "\n"; }
    void g() { cout << "g" << "\n"; }
};
 
class B : private A {
public:
    A::f;         //使用::运算符使基类中的成员成为公有的。注意函数名后不能有括号。(C++11标准不允许,使用using A::f)
    A::a;         //(C++11标准不允许,使用using A::a)
    using A::g;   //使用using语句使基类中的成员函数g成为类B中的公有成员,注意函数名后不能有括号。
};
int main() {
    B m;
  //m.b=1;        //错误,因为类B是以私有方式继承的,类A中的成员在类B中是私有的,这里不能访问私有成员。
    m.f();
    m.g();
    m.a = 1;
}

三.继承时的构造函数和析构函数问题
1. 在继承中,基类的构造函数构建对象的基类部分,派生类的构造函数构建对象的派生类部分。

2. 当创建派生类对象时先用派生类的构造函数调用基类的构造函数构建基类,然后再执行派生类构造函数构造派生类。
即先构造基类再构造派生类的顺序。执行析构函数的顺序与此相反。

3. 调用基类带参数的构造函数的方法:

在派生类的构造函数中使用初始化列表的形式就可以调用基类带参数的构造函数初始化基类成员,如B():A(int i){},类B是类A的派生类。

4. 派生类的构造函数调用基类的构造函数的方法为:

4.1 如果派生类没有显示用初始化列表调用基类的构造函数时,这时就会用派生类的构造函数调用基类的默认构造
函数,构造完基类后,才会执行派生类的构造函数函数体,以保证先执行基类构造函数再执行派生类构造函数
的顺序,如果基类没有默认构造函数就会出错。
4.2 如果派生类用 显示的初始化列表调用基类的构造函数时,这时就会检测派生类的初始化列表,当检测到显示调
用基类的构造函数时,就调用基类的构造函数构造基类,然后再构造派生类,以保证先执行基类构造函数再执
行派生类构造函数的顺序,如果基类没有定义派生类构造函数初始化列表调用的构造函数版本就会出错。

5. 如果在基类中没有定义默认构造函数,但定义了其他构造函数版本,这时派生类中定义了几个构造函数的不同版本,
这时只要派生类有一个构造函数没有显示调用基类中定义的构造函数版本就会发生错误,因为编译器会首先检查派
生类构造函数调用基类构造函数的匹配情况,如果发现不匹配就会出错,即使没有创建任何类的对象都会出错,而
不管这个派生类的对象有没有调用派生类的这个构造函数。比如:基类有一个构造函数版本A(int i)而没有定义默认
构造函数,派生类B,有这几个版本的构造函数B():A(4){},B(int i):A(5){},再有语句B(int i, int j){}没有显示调用
基类定义的构造函数而是调用基类的默认构造函数,如果创建了B m和语句B m(1)时都会提示没有可用的基类默认
构造函数可用的错误,虽然这时类B的对象m没有调用派生类B的带有两个形参的构造函数,但同样会出错。

6. 同样的道理,如果基类中定义了默认构造函数,却没有其他版本的构造函数,而这时派生类却显示调用了基类构造
函数的其他版本,这时就会出错,不管你有没有创建类的对象,因为编译器会先在创建对象前就检查构造函数的匹
配问题。

7. 派生类只能初始化他的直接基类。比如类C是类B的子类,而类B又是类A的子类,这时class C:public B{public:
B():A(){} };将会出错,该语句试图显示调用类B的基类类A的构造函数,这时会出现类A不是类C的基类的错误。

8. 继承中的复制构造函数和构造函数一样,基类的复制构造函数复制基类部分,派生类的复制构造函数复制派生类部
分。

9.派生类复制构造函数调用基类复制构造函数的方法为:A(const A& m):B(m){}其中B是基类,A是派生类。

10.如果在派生类中定义了复制构造函数而没有用初始化列表显示调用基类的复制构造函数,这时不管基类是否定义了
复制构造函数,这时出现派生类对象的复制初始化情况时就将调用基类中的默认构造函数初始化基类的成员变量,
注意是默认构造函数不是默认复制构造函数,如果基类没有默认构造函数就会出错。也就是说派生类的复制构造函
数的默认隐藏形式是B(const B& j):A(){}这里B是A的派生类,也就是说如果不显示用初始化列表形式调用基类的
复制构告函数时,默认情况下是用初始化列表的形式调用的是基类的默认构造函数。

11.当在派生类中定义了复制构造函数且显示调用了基类的复制构造函数,而基类却没有定义基类的复制构造函数时,
这时出现派生类对象的复制初始化情况就将调用基类中的默认复制构造函数初始化基类部分,调用派生类的复制构
造函数初始化派生类部分,因为复制构造函数只有一种形式,即A(const A& m){},比如当出现调用时A(const A&
m):B(m){}如果这时基类B没有定义复制构造函数,则该语句将会调用派生类A的默认复制构造函数。

12.如果基类定义了复制构造函数,而派生类没有定义时,则会调用基类的复制构造函数初始化基类部分,调用派生类
的默认复制构造函数初始化派生类部分。
 文章来源地址https://www.toymoban.com/news/detail-615894.html

到了这里,关于C++ | 继承(基类,父类,超类),(派生类,子类)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C++】 为什么多继承子类重写的父类的虚函数地址不同?『 多态调用汇编剖析』

    👀 樊梓慕: 个人主页  🎥 个人专栏: 《C语言》 《数据结构》 《蓝桥杯试题》 《LeetCode刷题笔记》 《实训项目》 《C++》 《Linux》《算法》 🌝 每一个不曾起舞的日子,都是对生命的辜负 本篇文章主要是为了解答有关多态的那篇文章那块的一个奇怪现象,大家还记得这张

    2024年02月19日
    浏览(27)
  • 【C++】面向对象编程(二)面向对象的编程思维:virtual虚拟调用、继承、protected成员、派生类与基类

    默认情形下,成员函数的解析都是编译时静态进行。如果要让成员函数的解析在程序运行时动态进行,需要在成员函数的声明前加上virtual: 虚函数的作用: 用基类的指针指向不同的派生类的对象时,基类指针调用其虚成员函数,会调用 真正指向对象的成员函数 ,而

    2024年02月07日
    浏览(35)
  • C# 父类实现的 IEquatable<T> 接口,重写的相等性判断函数都能被子类继承

    在父类重写的 Equals 方法,继承的 IEquatable 接口,重写的 == ,!= 运算符,重写的 GetHashCode 方法在子类中仍然是表现为被重写的状态。 现在让子类也实现 IEquatable 接口 此时代码提示如下 子类实现接口后,调用的默认就是子类实现的 Equals 方法。也可以用强制转换调用父类的

    2024年02月16日
    浏览(33)
  • Java 强制类型转换原理(父类转子类、子类转父类)

    在Java中,对象的强制转换(也称为类型转换)是将一个对象的引用转换为另一个类的引用,前提是这两个类之间存在继承或实现关系。强制转换可能会导致运行时异常,因为在转换的过程中,如果对象的实际类型与转换的目标类型不兼容,就会抛出ClassCastException异常。 分析

    2024年02月07日
    浏览(25)
  • C++的基类和派生类构造函数

    基类的成员函数可以被继承,可以通过派生类的对象访问,但这仅仅指的是普通的成员函数,类的构造函数不能被继承。构造函数不能被继承是有道理的,因为即使继承了,它的名字和派生类的名字也不一样,不能成为派生类的构造函数,当然更不能成为普通的成员函数。

    2024年02月10日
    浏览(39)
  • c++和QT子类调用父类方法

    c++调用方式 : 父类名::方法名 QT调用方式 : __super::方法

    2024年02月13日
    浏览(35)
  • C++笔记之基类指针动态地指向某一个子类情况列举

    code review!

    2024年02月12日
    浏览(38)
  • Java feign接口调用后返回子类,序列化子类反序列化只得到父类

    需要修改序列化方法 我存的时候放的子类,接收到却是只有父类的数据 feign默认使用jackson进行序列化,需要在父类上加上注解 @JsonTypeInfo(use = JsonTypeInfo.Id.CLASS) 在父类头上增加注解: https://blog.csdn.net/qq_39493105/article/details/126061848

    2024年02月14日
    浏览(31)
  • Java父类强制转换为子类的三种情况(推荐)

    Father f = new Father(); Son s = (Son)f;//出错 ClassCastException Father f = new Son(); Son s = (Son)f;//可以 f只能用父类的方法 s可以用子类的所有方法 Son s = new Son(); Father f = (Father)s;//可以 f只能用父类的方法

    2024年02月08日
    浏览(33)
  • 高程 | 继承与派生(c++)

    ⭐️ 概念 类的 继承 ,是新的类从已有类那里 得到已有的特性 。 类的 派生 ,从已有类 产生新类的过程 。 原有的类称为基类或父类,产生的新类称为派生类或子类。 直接参与派生出某类的基类称为直接基类,基类的基类甚至更高层的基类称为间接基类。 ⭐️ 语法 class

    2024年02月19日
    浏览(25)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包