派生类成员的访问属性:

C++继承方式总共分为以下几种:public、private、protected三种(它们直接影响到派生类的成员、及其对象对基类成员访问的规则)。
(1)public(公有继承):继承时保持基类中各成员属性不变,并且基类中private成员被隐藏。派生类的成员只能访问基类中的public/protected成员,而不能访问private成员;派生类的对象只能访问基类中的public成员。
(2)private(私有继承):继承时基类中各成员属性均变为private,并且基类中private成员被隐藏。派生类的成员也只能访问基类中的public/protected成员,而不能访问private成员;派生类的对象不能访问基类中的任何的成员。
(3)protected(保护性继承):继承时基类中各成员属性均变为protected,并且基类中private成员被隐藏。派生类的成员只能访问基类中的public/protected成员,而不能访问private成员;派生类的对象不能访问基类中的任何的成员。

由上表可知:

public继承:不改变基类成员的访问控制。
private继承:派生类所继承的基类成员的访问控制都变为private。
protected继承:基类中的private成员的访问控制不变,其余的都变为protected。
基类的 public成员被派生类继承,且在派生类中是可见的(visible in the derived class)。
基类的 private成员被派生类继承,但在派生类中是不可见的(not visible in the derived class)。

受保护成员 protected Members

以下面两个例子来说明受保护成员的可见性。

class BC
{
public:
void set_x( int a )
{
x = a;
}
protected:
int get_x( ) const
{
return x;
}
private:
int x;
};
class DC : public BC
{
public:
void add2()
{
int c=get_x();//可以执行
set_x(c+2);//可以执行
}
}
void g()
{
DC d;
d.get_x( );//在客户代码中不可见,不能够执行
}
class BC
{
protected:
int get_w( ) const;
//……
};
class DC : public BC
{
public:
int get_val( ) const
{
return get_w( );//派生类中可见,可以执行
}
void base_w( const BC& b ) const
{
cout << b.get_w( ) << endl;//客户代码中不可见,不能够执行
}
};

保护成员是专为继承机制而设的。
受保护成员(A protected member)仅在自己的类和其派生类中是可见的。

继承方式不会影响基类成员在派生类中的能见度。

名字隐藏 Name hiding

如果在派生类中添加了成员(数据、函数),其与基类的成员重名,本地成员(the local member)隐藏继承来的成员 ( hides the inherited member)。

以下面的代码为例:

class BC {
public:
void h( float );
}; class DC : public BC {
public:
void h( char[ ] );
};

DC继承自BC,其中BC含有void h(float)函数,DC中含有void h(char[])函数,这两个函数的签名不同,那么是否能够构成函数的重载呢?

对于下面的两行代码都能够执行?

void  f ( )
{
DC d1;
d1.h( "Boffo!" );//可以执行
d1.h( 707.7 );//不可以执行
}

其实这种想法是不对的,重载必须是同一级的函数才能构成,而这两个函数的级别是不一致的,本地成员void h(char[])将会隐藏继承来的成员 void h(float)!!

void  f ( )
{
DC d1;
d1.BC::h( 707.7 );//这样写是可以的
}

再就一个例子

对于一个实现数组升序的类继承自一个数组类。

class Array
{
public:
void insert(int X)
{
将X插入到 last_pos 指定的位置;
last_pos++;
}
private:
int last_pos;
//……
};
class AscArray : public Array
{
public:
void insert( int X )
{
确定插入的位置,并将X插入
// ...
}
// ...
};

其调用函数如下调用语法是正确的,但是其内涵错误

void  f (AscArray& as )
{
as.insert( );//正确,排序数组类中插入10然后实现排序
as.Array::insert();//不正确,使用了数组类的插入方法,将10插入到了数组尾部,并不能实现排序功能 }

调整可访问性 Adjusting access

一个继承成员的访问控制可能通过使用using声明( using declaration)改变。
还是上面那个例子。

class AscArray : public Array
{
private:
using Array::insert;
//将基类的public成员的外部访问权降低,使得无法通过派生类对象访问该成员!
public:
void insert( int X ) {
确定插入的位置,并将X插入
// ...
}
// ...
};
void f(AscArray& as )
{
as.insert( );//正确
as.Array::insert();//运行错误
}

在使用using声明时,基类中公有的成员在公共派生类中必须是公有的,只有这样才能保证公有继承时“派生类对象是一个基类对象”的逻辑关系。

在基类中的private成员,不能在派生类中任何地方用using声明。

同时,在基类中的protected成员,可在public派生下通过using声明改为public成员。

#include <iostream>
using namespace std;
class A
{
protected:
void PrintA( )
{
cout << "A::Print"<<endl ;
}
};
class B: public A
{
public:
using A::PrintA;//改为公有
public:
void PrintB( )
{
cout << "B::Print" <<endl;
}
};
int main( )
{
A a;
B b;
b.PrintB( );
b.PrintA( );
return ;
}

C++ 派生类成员的访问属性的更多相关文章

  1. C++:调整基类成员在派生类中的访问属性的其他方法(同名成员和访问声明)

    4.3 调整基类成员在派生类中的访问属性的其他方法 4.3.1 同名函数 在定义派生类的时候,C++语言允许在派生类中说明的成员与基类中的成员名字相同,也就是 说,派生类可以重新说明与基类成员同名的成 ...

  2. C++学习之路—继承与派生(一):基本概念与基类成员的访问属性

    (本文根据<c++程序设计>(谭浩强)总结而成,整理者:华科小涛@http://www.cnblogs.com/hust-ghtao,转载请注明) 1   基本思想与概念 在传统的程序设计 ...

  3. 3.3 C++改变基类成员在派生类中的访问属性

    参考:http://www.weixueyuan.net/view/6360.html 总结: 使用using声明可以改变基类成员在派生类中的访问属性. private: using book::se ...

  4. 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成员)

    [源码下载] 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成 ...

  5. python-面向对象(四)——类成员的访问方式汇总

    类成员的访问方式 #!/usr/bin/env python # _*_coding:utf-8 _*_ class pepole(object): '''This is __doc__ inform ...

  6. CPP/类/成员函数访问权限2

    // main.cpp // OOL // Created by mac on 2019/4/4. // Copyright © 2019年 mac. All rights reserved. // ...

  7. C++继承具体解释之二——派生类成员函数具体解释(函数隐藏、构造函数与兼容覆盖规则)

    在这一篇文章開始之前.我先解决一个问题. 在上一篇C++继承详解之中的一个--初探继承中,我提到了在派生类中能够定义一个与基类成员函数同名的函数,这样派生类中的函数就会覆盖掉基类的成员函数. 在谭浩强 ...

  8. C++——派生类中的访问——可见性问题

    C++中派生类对基类成员的访问形式主要有以下两种: 1.内部访问:由派生类中新增成员对基类继承来的成员的访问. 2.对象访问:在派生类外部,通过派生类的对象对从基类继承来的成员的访问.今天给大家介绍在 ...

  9. 在client类中设置访问属性 address,business和individua

    php 5.4中的traits,是新引入的特性,中文还真不知道如何准确翻译好.其实际的目的, 是为了有的场合想用多继承,但PHP又没多继承 ,于是就发明了这样的一个东西. Traits可以理解为一组能 ...

随机推荐

  1. Python中日期时间案例演示

    案例:准备10个人姓名,然后为这10个人随机生成生日[都是90后] 1.统计出那些人是夏季[6月-8月]出生的. 2.最大的比最小的大多少天 3.谁的生日最早,谁的生日最晚 备注:春季[3-5]夏季[ ...

  2. Flex 布局知识点梳理

    传统的布局方案,在针对特殊布局时会很不方便,比如垂直居中,把一个容器等分为N列等等.自从 Flex 出现以后,这些都迎刃而解了,本文对Flex相关内容做一个简单梳理. 什么是 Flex Flex 是 ...

  3. VS2012中使用SOS调试CLR

    之前看了<用WinDbg探索CLR世界>的一些列文章,发现SOS真的是一个非常好的调试.net的工具, 然后又惊喜的在http://blogs.msdn.com/b/marioheward ...

  4. 不要以为字段以transient修饰的话就一定不会被序列化

    1: 先阅读这边文章:http://www.importnew.com/21517.html 2:被transient修饰真的会被序列化吗? 反例:java.util.ArrayList中底层存储数组 ...

  5. CSS做一个Switch开关

    本文为博主原创,转载请注明出处. Switch开关: 根据需求可知,Switch开关只有两种选择,true或false.所以我们想到HTML的checkbox控件,用它来做. <input id ...

  6. bip39

    BIP: 39 (助记词) Layer: Applications Title: Mnemonic code for generating deterministic keys Author: Mar ...

  7. WorldWind源码剖析系列:插件类Plugin、插件信息类PluginInfo和插件编译器类PluginCompiler

    插件类Plugin是所有由插件编译器加载的插件子类的抽象父类,提供对插件的轻量级的访问控制功能. 插件信息类PluginInfo用来存储关于某个插件的信息的类,可以理解为对插件类Plugin类的进一步 ...

  8. JS获取客户端公网IP和IP地址

    网上解决方案 1.通过搜狐接口 获取方式如下: //网页端引入脚本 <script type="text/javascript" src="http://pv.so ...

  9. GrowingIO接入SDK简介

    安装使用文档逐步操作 准备工作: 1.注册一个GrowingIO账号 2.申请一个域名(注意:不能是ip或host) 登陆gio平台: 1.安装SDK 2.根据项目选择对应的sdk:js,安卓,ios ...

  10. centos发送邮件

    这里使用mailx发送. #yum -y install mailx 安装成功后,进入家目录编写配置文件.配置发送方的邮箱.密码.发送的服务器 #vi ~/.mailrc set from=hello ...