C++基类与派生类

2024-06-04 08:36
文章标签 c++ 基类 派生类

本文主要是介绍C++基类与派生类,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.派生类的定义

        派生类语法如下:


class <派生类名> : <继承方式> <基类名>
{
private:派生类新增成员;
protected:派生类新增成员;
public:派生类新增成员;
};

说明:
(1)class:类声明的关键字;
(2)派生类名:新(派生)类名;
(3)继承方式:指出继承的方式是什么。继承方式包含以下3种。
• public(公有继承方式)使用的最多;
• private(私有继承方式);
• protected(保护继承方式)。
用来说明对于从基类继承来的成员的访问权限,这个稍后介绍。

如果不写继承方式,则默认为 private(私有继承方式)。

(4)基类名:指明派生类是由哪个基类继承而来的。
(5)派生类的成员包括如下3种。
• 吸收基类成员:派生类继承了基类的除了构造函数和析构函数以外的全部数据成员和函数成员。
• 新增成员:增添新的数据成员和函数成员,体现了派生类与基类的不同和个性,是派生类对基类的发展。
• 对基类成员进行改造,包含两层含义:一是对基类成员的访问控制方式进行改造;二是定义与基类同名的成员,即同名覆盖。


class A
{
private:int m_a;
protected:int m_b;
public:int m_c;
};class B :public A //B是A的派生类
{
private:int m_d;
};int main()
{cout << "A的大小=" << sizeof(A) << ",B的大小=" << sizeof(B) << endl;return 0;
}

程序执行结果如下:

1.png

B的大小为16,说明它把A中的3个成员变量全部继承了。
下面的代码把继承方式由public改为private


class A
{
private:int m_a;
protected:int m_b;
public:int m_c;
};class B :private A //B是A的派生类,继承方式改为private
{
private:int m_d;
};int main()
{cout << "private继承" << endl;cout << "A的大小=" << sizeof(A) << ",B的大小=" << sizeof(B) << endl;return 0;
}

程序执行结果如下:

1.png

说明:

基类的所有数据成员都被派生类继承,和基类成员的访问权限无关及继承方式无关。(m_a,m_b,m_c)的访问权限不同

B类的内存图如下:

1.png


class Base//基类
{
public:void show(){cout << m_name << "," << m_a << endl;}
public:string m_name; //名字int m_a; //值
};class Derived :public Base //派生类继承了基类的成员
{
//什么都没有增加
};int main()
{Derived d;d.m_a = 10;d.m_name = "派生类";d.show();return 0;
}

1.png

从上面的执行结果可以看出,派生类虽然什么都没有写,但它继承了基类的成员数据(m_a和m_name)和成员函数(show)。

2. 继承方式和派生类的访问权限

派生类继承了基类的数据成员和成员函数,但是这些成员并不都能直接被派生类所访问。采用不同的继承方式,决定了基类成员在派生类中的访问权限。

在C++程序设计中,提供了3种继承方式:公有继承(public)、保护继承(protected)和私有继承(private)。访问权限public > protected > private 。下表为不同继承方式下,派生类对于从基类继承的成员的访问权限。
注意:protected不能在类外访问,它和private的区别是在继承时有区别,不继承时没有区别。

访问属性

继承方式

1.png


总结:1.private成员不能被派生类访问;2.权限按小的合并。

1.公有继承(public)

使用的最多的继承方式。
父类中的私有不能在子类中访问,公有还是公有,保护还是保护。

class A
{
public:void show1(){cout << "A::show1(),公有方法" << endl;}
protected:void show2(){cout << "A::show2(),保护方法" << endl;}
private:void show3(){cout << "A::show3(),私有方法" << endl;}
public:int m_a = 0;
protected:int m_b = 1;
private:int m_c = 2;
};
class B :public A
{
public:void show(){show1();//可以,继承下来是public,,可以在类外访问show2();//可以,继承下来是protected,不可以在类外访问//show3();//不能访问cout << m_a << endl;//可以,继承下来是public,可以在类外访问cout << m_b << endl;//可以,继承下来是protected,不可以在类外访问//cout << m_c << endl;//不能访问}
};int main()
{B bb;bb.show1();//public可以在类外访问//bb.show2();//错误,protected不能在类外访问bb.m_a = 100;//public可以在类外访问//bb.m_b = 200;//错误,protected不能在类外访问bb.show();//public可以在类外访问return 0;
}

1.png

说明:
(1).A中的show1,show2方法可以在子类B中被访问(第30,31行),但A中的私有方法show3在子类B中不能被访问(第32行)。
(2).show1在B中是公有,那么在类外也能被访问(第43行),但show2在B中是保护,在类外不能被访问(第44行)。
(3).A中的m_a,m_b可以在子类B中被访问(第34,35行)。但A中的私有成员m_c在子类B中不能被访问(第36行)。
(4).m_a在B中是公有,那么在类外也能被访问(第46行),但m_b在B中是保护,在类外不能被访问(第47行)。

2.保护继承(protected)

父类中的私有不能在子类中访问,公有变保护,保护还是保护。


//保护继承
class A
{
public:void show1(){cout << "A::show1(),公有方法" << endl;}
protected:void show2(){cout << "A::show2(),保护方法" << endl;}
private:void show3(){cout << "A::show3(),私有方法" << endl;}
public:int m_a = 0;
protected:int m_b = 1;
private:int m_c = 2;
};
class B :protected A
{
public:void show(){show1();//可以show2();//可以//show3();//不能访问cout << m_a << endl;//可以cout << m_b << endl;//可以//cout << m_c << endl;//不能访问}
};int main()
{B bb;//bb.show1();//错误,protected不能在类外访问//bb.show2();//错误,protected不能在类外访问//bb.m_a = 100;错误,protected不能在类外访问//bb.m_b = 200;//错误,protected不能在类外访问bb.show();//public可以在类外访问return 0;
}

1.png

3.私有继承(private)

父类的私有在子类中不能访问,公有变私有,保护变私有。


//私有继承
class A
{
public:void show1(){cout << "A::show1(),公有方法" << endl;}
protected:void show2(){cout << "A::show2(),保护方法" << endl;}
private:void show3(){cout << "A::show3(),私有方法" << endl;}
public:int m_a = 0;
protected:int m_b = 1;
private:int m_c = 2;
};
class B :private A
{
public:void show(){show1();//可以show2();//可以//show3();//不能访问cout << m_a << endl;//可以cout << m_b << endl;//可以//cout << m_c << endl;//不能访问}
};int main()
{B bb;//bb.show1();//错误,protected不能在类外访问//bb.show2();//错误,protected不能在类外访问//bb.m_a = 100;错误,protected不能在类外访问//bb.m_b = 200;//错误,protected不能在类外访问bb.show();//public可以在类外访问return 0;
}

1.png

说明:私有继承从上面看貌似和保护继承没有什么区别,实际上有区别,如果C再继承B,那么结果就不一样了。

这篇关于C++基类与派生类的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1029582

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现

c++的初始化列表与const成员

初始化列表与const成员 const成员 使用const修饰的类、结构、联合的成员变量,在类对象创建完成前一定要初始化。 不能在构造函数中初始化const成员,因为执行构造函数时,类对象已经创建完成,只有类对象创建完成才能调用成员函数,构造函数虽然特殊但也是成员函数。 在定义const成员时进行初始化,该语法只有在C11语法标准下才支持。 初始化列表 在构造函数小括号后面,主要用于给

2024/9/8 c++ smart

1.通过自己编写的class来实现unique_ptr指针的功能 #include <iostream> using namespace std; template<class T> class unique_ptr { public:         //无参构造函数         unique_ptr();         //有参构造函数         unique_ptr(