友元函数(Friend functions)

在前面的章节中我们已经看到了对class的不同成员存在3个层次的内部保护:public, protected 和 private。在成员为 protected 和 private的情况下,它们不能够被从所在的class以外的部分引用。然而,这个规则可以通过在一个class中使用关键字friend来绕过,这样我们可以允许一个外部函数获得访问class的protected 和 private 成员的能力。

为了实现允许一个外部函数访问class的private 和 protected 成员,我们必须在class内部用关键字friend来声明该外部函数的原型,以指定允许该函数共享class的成员。在下面的例子中我们声明了一个 friend 函数 duplicate:

    // friend functions#include <iostream.h>class CRectangle {int width, height;public:void set_values (int, int);int area (void) {return (width * height);}friend CRectangle duplicate (CRectangle);};void CRectangle::set_values (int a, int b) {width = a;height = b;}CRectangle duplicate (CRectangle rectparam) {CRectangle rectres;rectres.width = rectparam.width*2;rectres.height = rectparam.height*2;return (rectres);}int main () {CRectangle rect, rectb;rect.set_values (2,3);rectb = duplicate (rect);cout << rectb.area();}
24

函数duplicate是CRectangle的friend,因此在该函数之内,我们可以访问CRectangle 类型的各个object的成员 width 和 height。注意,在 duplicate()的声明中,及其在后面main()里被调用的时候,我们并没有把duplicate 当作class CRectangle的成员,它不是。

friend 函数可以被用来实现两个不同class之间的操作。广义来说,使用friend 函数是面向对象编程之外的方法,因此,如果可能,应尽量使用class的成员函数来完成这些操作。比如在以上的例子中,将函数duplicate() 集成在class CRectangle 可以使程序更短。

友元类 (Friend classes)

就像我们可以定义一个friend 函数,我们也可以定义一个class是另一个的friend,以便允许第二个class访问第一个class的 protected 和 private 成员。

    // friend class#include <iostream.h>class CSquare;class CRectangle {int width, height;public:int area (void) {return (width * height);}void convert (CSquare a);};Class CSquare {private:int side;public:void set_side (int a){side=a;}friend class CRectangle;};void CRectangle::convert (CSquare a) {width = a.side;height = a.side;}int main () {CSquare sqr;CRectangle rect;sqr.set_side(4);rect.convert(sqr);cout << rect.area();return 0;}
16

在这个例子中,我们声明了CRectangle 是CSquare 的friend,因此CRectangle可以访问CSquare 的protected 和 private 成员,更具体地说,可以访问CSquare::side,它定义了正方形的边长。

在上面程序的第一个语句里你可能也看到了一些新的东西,就是class CSquare空原型。这是必需的,因为在CRectangle 的声明中我们引用了CSquare (作为convert()的参数)。CSquare 的定义在CRectangle的后面,因此如果我们没有在这个class之前包含一个CSquare 的声明,它在CRectangle中就是不可见的。

这里要考虑到,如果没有特别指明,友元关系(friendships)并不是相互的。在我们的CSquare 例子中,CRectangle 是一个friend类,但因为CRectangle 并没有对CSquare作相应的声明,因此CRectangle 可以访问CSquare 的 protected 和private 成员,但反过来并不行,除非我们将 CSquare 也定义为CRectangle的 friend。

类之间的继承(Inheritance between classes)

类的一个重要特征是继承,这使得我们可以基于一个类生成另一个类的对象,以便使后者拥有前者的某些成员,再加上它自己的一些成员。例如,假设我们要声明一系列类型的多边形,比如长方形CRectangle或三角形CTriangle。它们有一些共同的特征,比如都可以只用两条边来描述:高(height)和底(base)。

这个特点可以用一个类CPolygon 来表示,基于这个类我们可以引申出上面提到的两个类CRectangle 和 CTriangle 。

类CPolygon 包含所有多边形共有的成员。在我们的例子里就是: width 和 height。而CRectangle 和 CTriangle 将为它的子类(derived classes)。

由其它类引申而来的子类继承基类的所有可视成员,意思是说,如果一个基类包含成员A ,而我们将它引申为另一个包含成员B的类,则这个子类将同时包含 A 和 B。

要定义一个类的子类,我们必须在子类的声明中使用冒号(colon)操作符: ,如下所示:

class derived_class_name: public base_class_name;

这里derived_class_name 为子类(derived class)名称,base_class_name 为基类(base class)名称。public 也可以根据需要换为protected 或 private,描述了被继承的成员的访问权限,我们在以下例子后会很快看到:

    // derived classes#include <iostream.h>Class CPolygon {protected:int width, height;public:void set_values (int a, int b) { width=a; height=b;}};class CRectangle: public CPolygon {public:int area (void){ return (width * height); }};class CTriangle: public CPolygon {public:int area (void){ return (width * height / 2); }};int main () {CRectangle rect;CTriangle trgl;rect.set_values (4,5);trgl.set_values (4,5);cout << rect.area() << endl;cout << trgl.area() << endl;return 0;}
20
10

如上所示,类 CRectangle 和 CTriangle 的每一个对象都包含CPolygon的成员,即: width, height 和 set_values()。

标识符protected 与 private类似,它们的唯一区别在继承时才表现出来。当定义一个子类的时候,基类的protected 成员可以被子类的其它成员所使用,然而private 成员就不可以。因为我们希望CPolygon的成员width 和 height能够被子类CRectangle 和 CTriangle 的成员所访问,而不只是被CPolygon自身的成员操作,我们使用了protected 访问权限,而不是 private。

下表按照谁能访问总结了不同访问权限类型:

可以访问 public protected private
本class的成员 yes yes yes
子类的成员 yes yes no
非成员 yes no no

这里"非成员"指从class以外的任何地方引用,例如从main()中,从其它的class中或从全域(global)或本地(local)的任何函数中。

在我们的例子中,CRectangle 和CTriangle 继承的成员与基类CPolygon拥有同样的访问限制:

   CPolygon::width           // protected accessCRectangle::width         // protected accessCPolygon::set_values()    // public accessCRectangle::set_values()  // public access

这是因为我们在继承的时候使用的是public,记得我们用的是:

class CRectangle: public CPolygon;

这里关键字 public 表示新的类(CRectangle)从基类(CPolygon)所继承的成员必须获得最低程度保护。这种被继承成员的访问限制的最低程度可以通过使用 protected 或 private而不是public来改变。例如,daughter 是mother 的一个子类,我们可以这样定义:

class daughter: protected mother;

这将使得protected 成为daughter 从mother处继承的成员的最低访问限制。也就是说,原来mother 中的所有public 成员到daughter 中将会成为protected 成员,这是它们能够被继承的最低访问限制。当然这并不是限制daughter 不能有它自己的public 成员。最低访问权限限制只是建立在从mother中 继承的成员上的。

最常用的继承限制除了public 外就是private ,它被用来将基类完全封装起来,因为在这种情况下,除了子类自身外,其它任何程序都不能访问那些从基类继承而来的成员。不过大多数情况下继承都是使用public的。

如果没有明确写出访问限制,所有由关键字class 生成的类被默认为private ,而所有由关键字struct 生成的类被默认为public。

什么是从基类中继承的? (What is inherited from the base class?)

理论上说,子类(drived class)继承了基类(base class)的所有成员,除了:

  • 构造函数Constructor 和析构函数destructor
  • operator=() 成员
  • friends

虽然基类的构造函数和析构函数没有被继承,但是当一个子类的object被生成或销毁的时候,其基类的默认构造函数 (即,没有任何参数的构造函数)和析构函数总是被自动调用的。

如果基类没有默认构造函数,或你希望当子类生成新的object时,基类的某个重载的构造函数被调用,你需要在子类的每一个构造函数的定义中指定它:

derived_class_name (parameters) : base_class_name (parameters) {}

例如 (注意程序中黑体的部分):

    // constructors and derivated classes#include <iostream.h>class mother {public:mother (){ cout << "mother: no parameters\n"; }mother (int a){ cout << "mother: int parameter\n"; }};class daughter : public mother {public:daughter (int a){ cout << "daughter: int parameter\n\n"; }};class son : public mother {public:son (int a) : mother (a){ cout << "son: int parameter\n\n"; }};int main () {daughter cynthia (1);son daniel(1);return 0;}
mother: no parameters
daughter: int parameter

mother: int parameter
son: int parameter

观察当一个新的daughter object生成的时候mother的哪一个构造函数被调用了,而当新的son object生成的时候,又是哪一个被调用了。不同的构造函数被调用是因为daughter 和 son的构造函数的定义不同:

   daughter (int a)          // 没有特别制定:调用默认constructorson (int a) : mother (a)  // 指定了constructor: 调用被指定的构造函数

多重继承(Multiple inheritance)

在C++ 中,一个class可以从多个class中继承属性或函数,只需要在子类的声明中用逗号将不同基类分开就可以了。例如,如果我们有一个特殊的class COutput 可以实现向屏幕打印的功能,我们同时希望我们的类CRectangle 和 CTriangle 在CPolygon 之外还继承一些其它的成员,我们可以这样写:

class CRectangle: public CPolygon, public COutput {
class CTriangle: public CPolygon, public COutput {

以下是一个完整的例子:

    // multiple inheritance#include <iostream.h>class CPolygon {protected:int width, height;public:void set_values (int a, int b){ width=a; height=b;}};class COutput {public:void output (int i);};void COutput::output (int i) {cout << i << endl;}class CRectangle: public CPolygon, public COutput {public:int area (void){ return (width * height); }};class CTriangle: public CPolygon, public COutput {public:int area (void){ return (width * height / 2); }};int main () {CRectangle rect;CTriangle trgl;rect.set_values (4,5);trgl.set_values (4,5);rect.output (rect.area());trgl.output (trgl.area());return 0;}
20
10

C++ 面向对象(三)—— 类之间的关系相关推荐

  1. Day-16 面向对象03 类与类之间的关系

    一.类与类之间的依赖关系 我用着你,但是你不属于我,这种关系是最弱的,比如,公司和雇员之间,对于正式员工,肯定要签订劳动合同,还得小心伺候着,但是如果是兼职,那无所谓,需要了你就来,不需要你就可以拜拜 ...

  2. 02 面向对象之:类空间问题以及类之间的关系

    一. 类的空间问题 1.1 何处可以添加对象属性 class A:def __init__(self,name):self.name = namedef func(self,sex):self.sex ...

  3. 详解:面向对象与面向过程的比较 类之间的关系:泛化、实现、依赖、关联、聚合、组合

    文章目录 1.面向对象程序设计概述 1.1 面向对象程序设计 1.2 传统结构化程序设计 1.3 面向对象与面向过程举例 2.类之间的关系 2.1 泛化/继承 2.2 实现 2.3 依赖 2.4 关联 ...

  4. 面向对象之: 类空间问题及类之间的关系

    面向对象之:类空间问题以及类之间的关系 类的空间问题 添加对象属性 总结:对象的属性不仅可以在__init__里面添加,还可以在类的其他方法或者类的外面添加. class A:mind = " ...

  5. 面向对象的基本概念(二)--UML.类之间的关系

    8.接口和类继承 (1)接口中不能有非抽象方法,但抽象类中可以有. (2)一个类能实现多个接口,但只能有一个父类. ( 3)接口并不属于继承结构,它实际与继承无关,因此无关的类也可以实现同一个接口. ...

  6. 【设计模式学习笔记】类图:类与类之间的关系

    目录 一.UML统一建模语言简介 二.类图 三.类与类之间的关系 1. 泛化关系 2. 实现关系 3.  依赖关系 4. 关联关系 5. 聚合关系 6. 组合关系 四.小结 一.UML统一建模语言简介 ...

  7. UML类图简介及类与类之间的关系

    UML(Unified Modeling Language,统一建模语言)建模是面向对象开发设计方法中的第一步,用UML来表达设计模式不仅方便了开发人员的交流,而且更加清晰.准确.UML定义了5类10 ...

  8. python类与类的关系_python 类与类之间的关系

    一.依赖关系(紧密程度最低) (1)简单的定义:就是方法中传递一个对象.此时类与类之间存在依赖关系,此关系比较低. (2)实例植物大战僵尸简易版 题目要求:创建一个植物,创建一个僵尸 1.植物:名字. ...

  9. 类与类之间的关系:依赖关系和关联关系及继承关系中self是什么? 类里面的特殊成员...

    类与类之间的关系 ⼤千世界, 万物之间皆有规则和规律. 我们的类和对象是对⼤千世界中的所有事物进行归 类. 那事物之间存在着相对应的关系. 类与类之间也同样如此. 在⾯向对象的世界中. 类与类 中存在 ...

  10. 信息系统项目管理师必背核心考点(四)UML类与类之间的关系

    科科过<每天一小时 俩月拿证>为您带来软考信息系统项目管理师核心重点考点(四):UML类与类之间的关系,内含思维导图+真题.本资料由科科过整理. [信息系统项目管理师核心考点]UML类与类 ...

最新文章

  1. Visual Studio 2005 Professional Released
  2. python如何小写p转换p_Python进阶---python 中字符串大小写转换
  3. Kafka 副本OffsetOutOfRangeException
  4. laravel框架的数据库链接
  5. html%2b怎么转换成加号,Apache mod_rewrite%2B和加号(+)符号
  6. Ionic简介和环境安装
  7. linux firefox 检查组件是否加载,利用火狐浏览器查看网站加载速度
  8. (筆記) 如何增加SignalTap II能觀察的reg與wire數量? (SOC) (Quartus II) (SignalTap II)
  9. 直击鲲鹏开发者训练营下一城,11.20 福州见
  10. c# mvc如何生成excel
  11. Ros平台下:从零开始学习SLAM(序)
  12. 蒙版操作—剪切蒙版制作艺术字
  13. 分析家数据格式、结构
  14. 全球最最可爱的的10种著名小型犬
  15. LeetCode第714题解析
  16. Floyd Thomas - Principles of Electric Circuits_ Conventional Current-Pearson (2021) 电路基础书籍推荐
  17. 一个账号多路由器拨号失败服务器无响应,PPPoE拨号失败 PPPoE拨号失败服务器无响应怎么办?...
  18. SQL Server代理(已禁用代理XP) 出现的原因以及解决方法【通俗易懂,简洁明了】
  19. stm32h743能跑linux吗,关于stm32H743 can配置
  20. CreateSpecificCulture('zh-cn')和new CultureInfo('zh-cn')的区别

热门文章

  1. iOS 添加导航栏两侧按钮
  2. 在屏幕上打印杨辉三角
  3. 怎样判断网页是静态还是伪静态呢
  4. 小女也爱c#(3)--俄罗斯方块练习数组
  5. 600. 不含连续1的非负整数
  6. 如何在24行JavaScript中实现Redux
  7. refract推导_我们如何利用Refract来利用React式编程的力量
  8. 如何免费注册Coursera课程
  9. InfluxDB的HTTP API写入操作
  10. Mysql添加字段.md