C++基础知识-派生类、调用顺序、访问等级、函数遮蔽
一、派生类的概念
类之间有一种层次关系,有父亲类,有孩子类。
车这个类,当成父类(也叫基类、超类),派生出卡车、轿车,他们属于孩子类(子类、派生类)
继承:有父亲类,有孩子类,构成了层次关系。继承这种概念,是咱们面向对象程序设计的核心思想之一。
我们通过继承父类来构建新的类:子类;所以,我们只需要写和子类相关的一些内容即可。
子类一般会比父类更加庞大。
二、派生类对象定义时调用构造函数的顺序
当定义子类对象时,是要调用父类和子类的构造函数的,而且父类的构造函数的函数体先执行,子类的构造函数的函数体后执行。
三、public、protected、private
三种访问权限 | 三种继承访问 | |
public:可以被任意实体所访问 | public继承 | |
protected:只允许本类或者子类的成员函数来访问 | protected继承 | |
private:只允许本类的成员函数来访问 | private继承 | |
基类中的访问权限 | 子类继承基类的继承方式 | 子类得到的访问权限 |
public | public | public |
protected | public | protected |
private | public | 子类无访问权限 |
public | protected | protected |
protected | protected | protected |
private | protected | 子类无访问权限 |
public | private | private |
protected | private | private |
private | private | 子类无访问权限 |
总结:
(1)子类public继承父类不改变父类的访问权限;
(2)protected继承父类中public成员变为子类protected成员;
(3)private继承使得父类中所有成员在子类中的访问权限变为private;
(4)对父类中的private成员不受继承方式的影响,子类永远无访问权限;
(5)对父类来讲,尤其是父类的成员函数,如果你不想让外界访问,就设置为private,如果你想让自己的子类能够访问,就设置为protected,如果你想公开,就设置为public。
四、函数遮蔽
子类中如果有一个同名函数,那么父类中,不管有几个同名函数,子类中都无法访问到。
如果我们确实下你给用父类中的同名函数,该怎么办?
(1)在子类的成员函数中,用“父类::函数名”轻质调用父类函数。
(2)using : using namespace
C++11中,让父类同名函数在子类中可见;
通过using这个关键字,让父类的同名函数在子类中可见,即“让父类同名函数在子类中以重载的方式来使用”。
说明:
(a)using Human::samenamefunc:只能指定函数名,则凡是基类中的public的samenamefunc,在子类中都可见。你无法让一部分父类中的同名函数不可见;
(b)using引入的主要目的是用来实现在子类对象中调用父类的重载版本。该函数在父类中的参数跟子类中的参数、类型、个数不同。
本节案例:
// Main.cpp
#include <iostream>
#include "Human.h"
#include "Men.h" using namespace std; int main()
{
Men men; // 当定义子类对象时,是要调用父类和子类的构造函数,父类的构造函数的函数体先执行,子类的构造函数的函数体后执行
men.samenfunc(,,,); // 调用子类本身的方法
men.samefunc(); // 因为在子类中声明了与父类同名的函数,所以不能使用父类的同名函数,但是根据using Human::samenamefunc使得子类可以使用父类的同名函数
men.samefunc();
men.samefunc(,);
men.samefunc(,,); return ;
} // Human.h
#ifndef __HUMAN__
#define __HUMAN__
#include <iostream>
class Human
{
public:
Human();
Human(int); public:
void samenamefunc();
void samenamefunc(int);
void samenamefunc(int,int);
void samenamefunc(int,int,int); public:
int m_Age; // 年龄
char m_Name[]; // 名字
};
#endif
// Human.cpp
#include "Human.h"
#include <iostream>
Human::Human()
{
std::cout << "调用Human::Human()" <<std::endl;
}
Human::Human(int)
{
std::cout << "调用Human::Human(int)" <<std::endl;
} void Human::samenamefunc()
{
Human::samenamefunc(); // 调用父类的方法
std::cout << "调用Human::samenamefunc()" <<std::endl;
}
void Human::samenamefunc(int)
{
std::cout << "调用Human::samenamefunc(int)" <<std::endl;
}
void Human::samenamefunc(int,int)
{
std::cout << "调用Human::samenamefunc(int,int)" <<std::endl;
}
void Human::samenamefunc(int,int,int)
{
std::cout << "调用Human::samenamefunc(int,int,int)" <<std::endl;
} // Men.h
#ifndef __MEN__
#define __MEN__
class Men : public Human // 表示Men是Human的子类
{
public:
Men(); public:
void samenamefunc(int,int,int,int); public:
using Human::samenamefunc; // 让父类的同名函数在子类中可见,即子类可以使用父类的同名函数
};
#endif // Men.cpp
#include "Men.h"
#include <iostream>
Men::Men()
{
std::cout << "调用了Men::Men()" << std::endl;
}
void Men::samenamefunc(int,int,int,int)
{
std::cout << "调用了Men::samenamefunc(int,int,int,int)" << std::endl;
}
C++基础知识-派生类、调用顺序、访问等级、函数遮蔽的更多相关文章
- Python基础知识:类
初级篇 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 面向对象:对函数进行分类和封装,让开发“更快更好更强...” 1.面向对象三大特性 ...
- C++:调整基类成员在派生类中的访问属性的其他方法(同名成员和访问声明)
4.3 调整基类成员在派生类中的访问属性的其他方法 4.3.1 同名函数 在定义派生类的时候,C++语言允许在派生类中说明的成员与基类中的成员名字相同,也就是 说,派生类可以重新说明与基类成员同名的成 ...
- 3.3 C++改变基类成员在派生类中的访问属性
参考:http://www.weixueyuan.net/view/6360.html 总结: 使用using声明可以改变基类成员在派生类中的访问属性. private: using book::se ...
- C++ 派生类成员的访问属性
派生类成员的访问属性: C++继承方式总共分为以下几种:public.private.protected三种(它们直接影响到派生类的成员.及其对象对基类成员访问的规则).(1)public(公有继承) ...
- 转 关于C#中派生类调用基类构造函数的理解
关于C#中派生类调用基类构造函数的理解 .c#class 本文中的默认构造函数是指在没有编写构造函数的情况下系统默认的无参构造函数 1. 当基类中没有自己编写构造函数时,派生类默认的调用 ...
- C++——派生类中的访问——可见性问题
C++中派生类对基类成员的访问形式主要有以下两种: 1.内部访问:由派生类中新增成员对基类继承来的成员的访问. 2.对象访问:在派生类外部,通过派生类的对象对从基类继承来的成员的访问.今天给大家介绍在 ...
- c++学习笔记4,调用派生类的顺序构造和析构函数(一个)
测试源代码: //測试派生类的构造函数的调用顺序何时调用 //Fedora20 gcc version=4.8.2 #include <iostream> using namespace ...
- Python基础:新式类的属性访问
一.概述 二.准备工作 1.讨论对象 2.名词解释 三.实例绑定的属性访问 1.获取属性 一般规则 参考源码 示例验证 2.设置属性 一般规则 参考源码 示例验证 3.删除属性 一般规则 参考源码 示 ...
- 关于C#中派生类调用基类构造函数的理解
(1)当基类中没有自己编写的构造函数时,派生类默认条用基类的构造函数 (2)当基类中有自己编写的构造函数时,要在基类中添加无参的构造函数 public class MyBaseClass { publ ...
随机推荐
- jsp滚动框(非滚动条)
<marquee scrollAmount=4 width=300>需要滚动的字</marquee> scrollAmount表示运动速度,值是正整数,默认为6,越大滚动越快 ...
- Template7学习记录
来源:http://idangero.us/template7/#.V2iXqJGF6Ul 测试用json数据: var jsonData = { people: [ { firstName: 'Jo ...
- c语言蛋疼的字符串赋值
我觉得c语言比较蛋疼的一个地方就是给字符串赋值,不是初始化,是赋值. char string[20]={0}; 你不能通过 string="hello";这种方式赋值.但是在字符串 ...
- __imp___vsnprintf
unresolved external symbol __imp___vsnprintf 解决方案找到了. 在vs2015工程选项,链接器附加依赖项里面添加legacy_stdio_definitio ...
- docker 构建 spring boot项目
在docker 开始部署springBoot项目 1.在centos7 ~ 创建一个文件夹docker 里面放置 上面的Dockerfile 和 springBoot 打包的项目docker_spri ...
- 前端传递对象列表到WebApi
public Int64 objectPOC(JObject jsonWrapper) { dynamic jsonValues = jsonWrapper; JArray jsonInput = j ...
- 看图说说JVM老年代垃圾收集器
注:G1垃圾收集器是目前最前沿的GC收集器,未来将取代CMS垃圾收集器,可以作为整个Heap的收集器使用,不限于老年代!!!
- ZOJ2748 Free Kick 2017-04-18 20:40 40人阅读 评论(0) 收藏
Free Kick Time Limit: 2 Seconds Memory Limit: 65536 KB In a soccer game, a direct free kick is ...
- Java程序中做字符串拼接时可以使用的MessageFormat.format
Java里从来少不了字符串拼接的活,Java程序员也肯定用到过StringBuffer,StringBuilder,以及被编译器优化掉的+=.但这些都和下文要谈的无关. 比如有这样的字符串: 张三将去 ...
- Objective-C 学习笔记(五) 指针
Objective-C 指针 每一个变量是一个内存位置和每一个存储单元都有其定义的地址,可以使用符号(&)的运算符,它表示内存中的地址访问. a. 我们定义一个指针变量 b. 分配一个指针变量 ...