• 公有继承(public)

公有继承在C++中是最常用的一种继承方式,我们先来看一个示例:

 #include<iostream>
using namespace std;
class Father{
public:
Father()=default;
void Father_show1(){
cout<<"调用Father类的public方法:Father_show1"<<endl;
}
protected:
void Father_show2(){
cout<<"调用Father类的protected方法:Father_show2"<<endl;
}
private:
void Father_show3(){
cout<<"调用Father类的private方法:Father_show3"<<endl;
}
}; class Son:public Father{
public:
Son()=default;
void Son_fun1(){
cout<<"调用Son类的public方法:Son_fun1"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
protected:
void Son_fun2(){
cout<<"调用Son类的protected方法:Son_fun2"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
private:
void Son_fun3(){
cout<<"调用Son类的private方法:Son_fun3"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误:无法调用Father类的private方法
}
}; int main(){
Son s;
s.Son_fun1(); //正确,只能调用对象的public方法
s.Father_show1();
//s.Son_fun2(); //错误:不能调用对象的protected方法
//s.Father_show2();
//s.Son_fun3(); //错误:不能调用对象的private方法
//s.Father_show3();
return ;
}

对公有继承的理解:

1.三种属性能力的强弱:public<protected<private

2.在C++的继承中,子类会继承父类中除构造函数和析构函数之外的所有成员(正所谓儿子无法继承父亲的生死) 。而公有继承(public)就相当于先将从父类那里继承的全部成员放到子类的public部分,如下:

 class Son:public Father{
2 /* 从Father类中继承的所有成员
3 public:
4 public:
5 void Father_show1(){
6 cout<<"调用Father类的public方法:Father_show1"<<endl;
7 }
8 protected:
9 void Father_show2(){
10 cout<<"调用Father类的protected方法:Father_show2"<<endl;
11 }
12 private:
13 void Father_show1(){
14 cout<<"调用Father类的public方法:Father_show1"<<endl;
15 }
16 */

public:
Son()=default;
void Son_fun1(){
cout<<"调用Son类的public方法:Son_fun1"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误:无法调用Father类的private方法
}
protected:
void Son_fun2(){
cout<<"调用Son类的protected方法:Son_fun2"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误: 无法调用Father类的private方法
}
private:
void Son_fun3(){
cout<<"调用Son类的private方法:Son_fun3"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误: 无法调用Father类的private方法
}
};

 然后根据三种属性能力的强弱决定成员的属性在子类中究竟是public、protected还是private:

• Father_show1():在Father类中属于public方法,继承到子类Son后放在类的public部分,由于public=public,因此在子类Son中Father_show1()方法仍是public方法

• Father_show2():在Father类中属于protected方法,继承到子类Son后放在类的public部分,由于protected>public,因此子类Son中Father_show2()方法是protected方法

• Father_show3():在Father类中属于private方法,可以理解为“父亲的隐私”,继承到子类Son后放在类的public部分,由于private>public,因此子类Son中Father_show3()方法是private方法。然而正所谓“儿子即使继承了父亲的财产,也无法知晓父亲的隐私”,因此不管儿子以何种方式(public/protected/private)继承父亲的“财产”也无法利用父亲的“隐私”去进行“交易”,换句话说就是父类的private成员虽然可以被子类继承,但子类中的任何成员方法都不能在其函数体中调用这些从父类中继承而来的private成员。因此Son类中的成员方法不管其为与什么部分,都无法调用Father_show3

3.对象只能调用其public部分的成员而不能调用protected和private部分的成员。因此上例中Son类的对象s可以调用方法Son_fun1()和方法Father_show1(),而无法调用方法Son_fun2()、Son_fun3()、Father_show2()和Father_show3()


• 保护继承(protected)

将上面的示例改为保护继承:

 #include<iostream>
using namespace std;
class Father{
public:
Father()=default;
void Father_show1(){
cout<<"调用Father类的public方法:Father_show1"<<endl;
}
protected:
void Father_show2(){
cout<<"调用Father类的protected方法:Father_show2"<<endl;
}
private:
void Father_show3(){
cout<<"调用Father类的private方法:Father_show3"<<endl;
}
}; class Son:protected Father{
public:
Son()=default;
void Son_fun1(){
cout<<"调用Son类的public方法:Son_fun1"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
protected:
void Son_fun2(){
cout<<"调用Son类的protected方法:Son_fun2"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
private:
void Son_fun3(){
cout<<"调用Son类的private方法:Son_fun3"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
}; int main(){
Son s;
s.Son_fun1(); //正确,只能调用对象的public方法
//s.Son_fun2(); //错误:不能调用对象的protected方法
//s.Father_show1();
//s.Father_show2();
//s.Son_fun3(); //错误:不能调用对象的private方法
//s.Father_show3();
return ;
}

 对保护继承的理解:

1.三种属性能力的强弱:public<protected<private

2.保护继承相当于先将从父类继承的所用成员都放在子类的protected部分:

 class Son:public Father{
/*
3 protected:
4 public:
5 void Father_show1(){
6 cout<<"调用Father类的public方法:Father_show1"<<endl;
7 }
8 protected:
9 void Father_show2(){
10 cout<<"调用Father类的protected方法:Father_show2"<<endl;
11 }
12 private:
13 void Father_show1(){
14 cout<<"调用Father类的public方法:Father_show1"<<endl;
15 }
16 */

public:
Son()=default;
void Son_fun1(){
cout<<"调用Son类的public方法:Son_fun1"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误:无法调用Father类的private方法
}
protected:
void Son_fun2(){
cout<<"调用Son类的protected方法:Son_fun2"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误: 无法调用Father类的private方法
}
private:
void Son_fun3(){
cout<<"调用Son类的private方法:Son_fun3"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误: 无法调用Father类的private方法
}
};

然后和公有继承一样,根据三种属性能力的强弱决定成员的属性在子类中究竟是public、protected还是private:

• 由于public<protected,因此方法Father_show1()在类Son中是protected方法

• 由于protected=protected,因此方法Father_show2()在类Son中是protected方法

• 就像在公有继承中分析的那样,Father_show3()在类Son中虽然是private方法,但Son类中的任何成员方法都不能在其函数体中调用方法Father_show3()

3.对象只能调用public部分的成员,此时方法Father_show1()是对象的protected方法,因此无法像公有继承那样再被显式调用了


• 私有继承

将上面的示例改为私有继承:

 #include<iostream>
using namespace std;
class Father{
public:
Father()=default;
void Father_show1(){
cout<<"调用Father类的public方法:Father_show1"<<endl;
}
protected:
void Father_show2(){
cout<<"调用Father类的protected方法:Father_show2"<<endl;
}
private:
void Father_show3(){
cout<<"调用Father类的private方法:Father_show3"<<endl;
}
}; class Son:private Father{
public:
Son()=default;
void Son_fun1(){
cout<<"调用Son类的public方法:Son_fun1"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
protected:
void Son_fun2(){
cout<<"调用Son类的protected方法:Son_fun2"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
private:
void Son_fun3(){
cout<<"调用Son类的private方法:Son_fun3"<<endl;
Father_show1();
Father_show2();
//Father_show3(); //错误:无法调用Father类的private方法
}
}; int main(){
Son s;
s.Son_fun1(); //正确,只能调用对象的public方法
//s.Son_fun2(); //错误:不能调用对象的protected方法
//s.Son_fun3(); //错误:不能调用对象的private方法
//s.Father_show1();
//s.Father_show2();
//
s.Father_show3();
return ;
}

对私有继承的理解:

1.三种属性能力的强弱:public<protected<private

2.私有继承相当于先将从父类继承的所用成员都放在子类的private部分:

 class Son:public Father{
2 /*
3 private:
4 public:
5 void Father_show1(){
6 cout<<"调用Father类的public方法:Father_show1"<<endl;
7 }
8 protected:
9 void Father_show2(){
10 cout<<"调用Father类的protected方法:Father_show2"<<endl;
11 }
12 private:
13 void Father_show1(){
14 cout<<"调用Father类的public方法:Father_show1"<<endl;
15 }
16 */

public:
Son()=default;
void Son_fun1(){
cout<<"调用Son类的public方法:Son_fun1"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误:无法调用Father类的private方法
}
protected:
void Son_fun2(){
cout<<"调用Son类的protected方法:Son_fun2"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误: 无法调用Father类的private方法
}
private:
void Son_fun3(){
cout<<"调用Son类的private方法:Son_fun3"<<endl;
Father_show1();
Father_show2();
//Father_show3();//错误: 无法调用Father类的private方法
}
};

然后和公有继承一样,根据三种属性能力的强弱决定成员的属性在子类中究竟是public、protected还是private:

• 由于public<private,因此方法Father_show1()在类Son中是private方法,但类Son中的成员方法可以在函数体内调用该方法

• 由于private>protected,因此方法Father_show2()在类Son中是prijvate方法,但类Son中的成员方法可以在函数体内调用该方法

• 就像在公有继承中分析的那样,Father_show3()在类Son中虽然是private方法,但Son类中的任何成员方法都不能在其函数体中调用方法Father_show3()

3.对象只能调用public部分的成员,此时方法Father_show1()是对象的private方法,因此无法像公有继承那样再被显式调用了


QUESTION:保护继承(protected)和私有继承(private)有何不同?

ANSWER:在上面的例子中,我们发现保护继承方式和私有继承方式达到的效果完全一样,难道这两中继承方式没有任何区别吗?我们先来看一个例子:

 #include<iostream>
using namespace std;
3 class GrandFather{ //祖父类
4 public:
5 GrandFather()=default;
6 void GrandFather_show(){
7 cout<<"调用GrandFather类的方法:GrandFather_show"<<endl;
8 }
9 };
10 class Father:protected GrandFather{ //父类
11 public:
12 Father()=default;
13 };
14 class Son:public Father{ //子类
15 public:
16 Son()=default;
17 void Son_show(){
18 cout<<"调用Son类的方法:Son_show"<<endl;
19 GrandFather_show();
20 }
21 };

int main(){
Son s;
s.Son_show();
return ;
}

我们发现上面的程序可以顺利运行。这是因为当Father类以保护方式(protected)继承GrandFather类时,GrandFather类中的公有方法GrandFather_show()会以protected方法的形式存在于类Father中,当类Son再以公有方式(public)继承类Father时,方法GrandFather_show()会仍以protected方法的形式存在与类Son中,由于一个类中的成员方法允许在其函数体内调用protected部分的成员,因此系统允许在Son类的成员方法Son_show()调用方法GrandFather_show(),从而使程序顺利运行。

现在我们将程序改为Father类以私有继承的方式继承GrandFather类:

 #include<iostream>
using namespace std;
3 class GrandFather{ //祖父类
4 public:
5 GrandFather()=default;
6 void GrandFather_show(){
7 cout<<"调用GrandFather类的方法:GrandFather_show"<<endl;
8 }
9 };
10 class Father:private GrandFather{ //父类
11 public:
12 Father()=default;
13 };
14 class Son:public Father{ //子类
15 public:
16 Son()=default;
17 void Son_show(){
18 cout<<"调用Son类的方法:Son_show"<<endl;
19 GrandFather_show();
20 }
21 };

int main(){
Son s;
s.Son_show();
return ;
}

我们发现程序报错。这是因为当Father类以私有(private)继承GrandFather类时,GrandFather类中的公有方法GrandFather_show()会以private方法的形式存在于类Father中,换句话说方法GrandFather_show()变成了类Father的“隐私”;当类Son再以公有方式(public)继承类Father时,由于“儿子无法利用父亲的“隐私”进行交易”,因此无法在Son类中的任何成员方法中调用GrandFather_show()方法,包括Son_show()。此时如果我们将类Son中成员函数Son_show()中的语句“GrandFather();"注释掉,程序便可以重新顺利执行。

 #include<iostream>
using namespace std;
3 class GrandFather{ //祖父类
4 public:
5 GrandFather()=default;
6 void GrandFather_show(){
7 cout<<"调用GrandFather类的方法:GrandFather_show"<<endl;
8 }
9 };
10 class Father:private GrandFather{ //父类
11 public:
12 Father()=default;
13 };
14 class Son:public Father{
15 public:
16 Son()=default;
17 void Son_show(){
18 cout<<"调用Son类的方法:Son_show"<<endl;
19 //GrandFather_show();
20 }
21 };

int main(){
Son s;
s.Son_show();
return ;
}


•总结

父类中的访问属性 继承方式 子类中的访问属性
private public/protected/private 不允许访问
public public public
public protected protected
public private private
protected public protected
protected protected protected
protected private private

C++:继承访问属性(public/protected/private)的更多相关文章

  1. C++继承中的public/protected/private

    今天杨老师讲到C++的继承的时候用一个表来说明子类继承父类后访问权限的变化,如下表: 注:在本类中,protected与private是相同的,但protected可以被继承,而private却不行. ...

  2. c/c++ 继承与多态 继承中的public, protected, private

    问题:类B私有继承类A,类A有个protected成员,那么在类B的成员函数里是否可以使用类A的protected成员? 可以使用. 估计有的同学说不对吧,类B都私有继承了类A了,怎么还能访问类A的p ...

  3. [学习笔记]Java的public,protected,private,缺省的作用域

    0.引言 Java的访问指示符public,protected,private,缺省可以用来修饰类和方法. 1.作用域如下 具体如下: 作用域       当前类    同一package   子孙类 ...

  4. C++中public,protected,private派生类继承问题和访问权限问题

    C++中public,protected,private派生类继承问题和访问权限问题 当一个子类从父类继承时,父类的所有成员成为子类的成员,此时对父类成员的访问状态由继承时使用的继承限定符决定. 1. ...

  5. 【转载】C++中public,protected,private访问

    第一:private, public, protected 访问标号的访问范围. 假如我们约定: 类内部-----指的是当前类类型的定义中,以及其成员函数的声明和定义中: 类外部-----指的是不在当 ...

  6. php public protected private属性实例详解

    php 类中函数和类变量都有三个属性:public protected private,具体什么时候使用什么属性好纠结,特意找了个实例,这样看起来更清晰. public 表示全局,类内部外部子类都可以 ...

  7. 【转】C++易混知识点5:实例讲解Public Protected Private作用域,继承的区别和用意

    大学生涯,涉及到类的作用域,继承都是用的public 共有继承,当时也没想那么多,觉得共有继承多方便,多简单,反正没有太多的限制,不管是类的成员或者是基类的成员函数都可以访问.没有深究.其实这里面真是 ...

  8. C# 成员默认访问权限(public、private、protected、internal)

    C# 成员默认访问权限(public.private.protected.internal) 来源 https://www.cnblogs.com/yezongjie/p/20181121Access ...

  9. C++ 类访问控制(public/protected/private)

    第一:private, public, protected 访问标号的访问范围. private:只能由1.该类中的函数.2.其友元函数访问. 不能被任何其他访问,该类的对象也不能访问. protec ...

随机推荐

  1. 9.算法之顺序、二分、hash查找

    一.查找/搜索 - 我们现在把注意力转向计算中经常出现的一些问题,即搜索或查找的问题.搜索是在元素集合中查找特定元素的算法过程.搜索通常对于元素是否存在返回 True 或 False.有时它可能返回元 ...

  2. C语言程序设计II—第一周教学

    第一周教学总结(25/2-3/3) 教学内容 开学谈心 测验数据类型.运算符与表达式的自学情况,并讲解测验题目 第七章 数组 7.1 一维数组 课前准备 在蓝墨云发布资源:回顾数据类型与表达式测试活动 ...

  3. cloudstack网络部分知识点汇总

    UI界面的几个网络选项 DNS1:供此区域的来宾VM使用,此区域的公用IP必须路由到此服务器: 内DNS:供此区域的系统VM使用,提供点的专用IP必须路由到此服务器: 如:你在添加存储设备时,用的主机 ...

  4. STM32 中 BIT_BAND(位段/位带)和别名区使用入门(转载)

    一. 什么是位段和别名区 是这样的,记得MCS51吗? MCS51就是有位操作,以一位(BIT)为数据对象的操作,MCS51可以简单的将P1口的第2位独立操作: P1.2=0;P1.2=1 :这样就把 ...

  5. jqgrid 配置分页大小及下拉项

    如何配置分页大小的下拉项?显示效果如下: 通过 rowNum 来设置默认分页大小 通过 rowList 来设置分页下拉.   rowList 的值为一个数组,比如:[10,20,30] $(" ...

  6. jqgrid 将列头设置为超链接或按钮

    有时,需要将某个列头设置为超链接或按钮,点击超链接或按钮能够跳转至其他页面(或执行一个事件操作). 可以把 label 值设置成一个a标签或button 代码如下: colModel: [{ labe ...

  7. 在AspNetCore 中 使用Redis实现分布式缓存 (转载)

    文章概念描述 分布式缓存描述:分布式缓存重点是在分布式上,相信大家接触过的分布式有很多中,像分布式开发,分布式部署,分布式锁.事物.系统 等有很多.使我们对分布式本身就有一个很明确的认识,分布式就是有 ...

  8. Iframe和Frame中实现cookie跨域的方法(转载)

    在Iframe和Frame中默认是不支持Cookie跨域的,但通过设置P3P协议相关的响应头可以解决这一问题.关于p3p协议: P3P: Platform for Privacy Preference ...

  9. WPF编程,通过Path类型制作沿路径运动的动画一种方法。

    原文:WPF编程,通过Path类型制作沿路径运动的动画一种方法. 版权声明:我不生产代码,我只是代码的搬运工. https://blog.csdn.net/qq_43307934/article/de ...

  10. 28 个 C/C++ 开源 JSON 程序库性能及标准符合程度评测

    28 个 C/C++ 开源 JSON 程序库性能及标准符合程度评测 坊间有非常多的 C/C++ JSON 库,怎么选择是一个难题. [nativejson-benchmark](https://git ...