3.5类的组合

Part1.应用背景

对于复杂的问题,往往可以逐步划分为一系列稍微简单的子问题。

解决复杂问题的有效方法是将其层层分解为简单的问题组合,首先解决简单问题复杂问题也就迎刃而解了。

在面向对象的程序设计中,可以对复杂对象进行分解、抽象,把一个复杂对象分解为简单对象的组合,由比较容易理解和实现的部件对象装配而成。

Part2.定义及代码

①定义:类的组合描述的就是一个雷内嵌其他类的对象作为成员的情况,他们之间的关系是一种包含与被包含的关系。

②作用机制:当创建类的对象时,如果这个类具有内嵌对象成员,那么各个内嵌对象将首先被自动创建。

      在创建对象时既要对本类的基本类型数据成员进行初始化又要对内嵌对象成员进行初始化。

③一般形式:

类名::类名(形参表):内嵌对象1(形参表),内嵌对象2(形参表),......
//内嵌对象1(形参表),...称作 初始化列表:对内嵌对象进行初始
{
类的初始化
}

④组合类的构造函数、析构函数调用顺序

构造函数调用顺序:

        1)调用内嵌对象的构造函数,调用顺序按照内嵌对象在组合类的定义中出现的次序。

         注意:内嵌对象在构造函数的初始化类表中出现的顺序与内嵌对象构造函数的调用顺序无关。

        2)执行本类构造函数的函数体。

析构函数调用顺序:

        析构函数的调用执行顺序与构造函数刚好相反。

例:

#include<iostream>
using namespace std;
class Mammal{
public:
Mammal()
{
cout << "Constructing Mammal." << endl;
}
~Mammal()
{
cout << "Desstructing Mammal." << endl;
}
};
class Dog :public Mammal {
public:
Dog()
{
cout << "Constructing Dog." << endl;
}
~Dog()
{
cout << "Desstructing Dog." << endl;
}
};
void main()
{
Dog d;
}

 ⑤组合类的复制构造函数

//类的组合,线段(Line)类
//使用一个类来描述线段,Point类的对象来表示端点,使Line类包括Point类的两个对象p1,p2,作为其数据成员。
//Line类具有计算线段长度的功能,在构造函数中实现
#include<iostream>
#include<cmath>
using namespace std; class Point {
public:
Point(int xx = , int yy = ) {//构造函数
x = xx;
y = yy;
}
Point(Point& p);//复制构造函数
int getX() { return x; }
int getY() { return y; }
private:
int x, y;
};
Point::Point(Point& p) {//复制构造函数的实现
x = p.x;
y = p.y;
cout << "Calling the copy constructor of Point" << endl;
}
//类的组合
class Line {
public:
Line(Point xp1, Point xp2);
Line(Line& l);
double getLen() { return len; }
private:
Point p1, p2;//Point类的对象p1,p2
double len;
};
//组合类的构造函数
Line::Line(Point xp1, Point xp2) :p1(xp1), p2(xp2) {
cout << "Calling constructor of Line" << endl;
double x = static_cast<double>(p1.getX() - p2.getX());
double y = static_cast<double>(p1.getY() - p2.getY());
len = sqrt(x * x + y * y);
}
//组合类的复制构造函数
Line::Line(Line& l) :p1(l.p1), p2(l.p2) {
cout << "Calling the copy constructor of Line" << endl;
len = l.len;
}
//主函数
int main()
{
Point myp1(, ), myp2(, );//建立Point类的对象
Line line(myp1, myp2);//建立Line类的对象
Line line2(line);//利用复制构造函数建立一个新对象
cout << "The length of the line is:";
cout << line.getLen() << endl;
cout << "The length of the line2 is:";
cout << line2.getLen() << endl;
return ;
}

分析:

  主程序执行时,首先生成连个Point类的对象,然后构造Line类的对象line,接着通过复制构造函数建立Line类的第二个对象line2,最后输出两点的距离。

  整个过程中,Point类的复制构造函数被调用了六次,而且都是在Line类构造函数体运行之前进行的,他们分别是两个对象在Line构造函数进行函数参数形实结合时,初始化内嵌对象时,以及复制构造函数line2时被调用的。两点的距离在Line类的构造函数中求得,存放在其私有数据成员len中,只能通过成员函数getLen()来访问。

  

3.6前向引用声明

Part1.应用场景

C++的类应该先定义再使用,但是在处理相对复杂的问题是,考虑类的组合时,很有可能遇到两个类相互引用的情况,这种情况也称为循环依赖。

因此,无论将哪一个类的定义放在前面,都会引起编译错误。

解决这种问题的方法,就是使用前向引用声明。

Part2.定义及代码

①定义:前向引用声明,是在引用未定义的类之前,将该类的名字告诉编译器,使编译器根据知道那是一个类名。

②例

class B;//前向引用声明
class A{//A类的定义
public:
void f(B b);//以B类对象b为形参的成员函数
};
class B{//B类的定义
public:
void g(A a);//以A类对象a为形参的成员函数
};

③注意:前向引用声明不是万能的。

class Frred;//前向引用声明
class Barney{
Fred x;//错误:类Fred的声明尚不完善,不能定义类Fred的数据成员
};
class Fred{
Barney y;
}

错误:对类的前向引用声明只能说明Fred是一个类名,而不能给出该类的完整定义,因此在类Barney中就不能定义类Fred的数据成员。

class Frred;//前向引用声明
class Barney{
public:
...
void method(){//错误:Fred类的对象在定义前被使用
x.yabbaDabbaDo;
}
private:
Fred &x;//正确:经过前向引用声明,可以声明Fred类的对象引用或指针
};
class Fred{
public:
...
void yaabaDabbaDo();
private:
Barney &y;
};

错误:编译时指出错误,因为在类Barney的内联函数中使用了有x所指向的、Fred类的对象,而此时Fred类尚未被完整的定义。

解决方法:更改这两个类的定义次序,或者将函数method()改为非内联形式,并且在类Fred的完整定义之后,再给出函数定义。

 

C++类的组合、前向引用声明的更多相关文章

  1. vs2015类中方法前的引用链接不显示的解决方案

    在工具→选项,打开如下界面,寻找“文本编辑器→所有语言”中设置显示:&lt;img data-rawheight="761" data-rawwidth="130 ...

  2. C++中对类的提前引用声明注意事项

    //或许,友元是VC++6.0心里永远的痛,对于这个BUG我一直很介意.//注:这个程序在VC++6.0里是行不通的,在VS2008里是可以的.#include <iostream> #i ...

  3. vs2017如何设置类或函数前不显示引用的数量

    这几天,从vs2013换成vs2017,17版本增加了一个类或函数前提示引用的数量,这个感觉很别扭,如何取消显示这个呢? 问题如下: 取消显示这个引用的步骤: 找到菜单栏: 工具 ---> 选项 ...

  4. vs2015如何设置类或函数前不显示引用的数量

    这几天,从vs2012换成vs2015,感觉15版本增加了一个类或函数前提示引用的数量,这个感觉很别扭,如何取消显示这个呢? 问题如下: 取消显示这个引用的步骤: 找到菜单栏: 工具 ---> ...

  5. 类声明、类作用域、前向声明、this指针、嵌套类、PIMPL 技法 等

    一.类声明 //类是一种用户自定义类型,声明形式: class 类名称 {    public:              公有成员(外部接口)    private:              私有 ...

  6. c++基础语法 构造函数 析构函数 类的组合

    1 构造函数 1.不能指定任何返回值,甚至连void都不能有. 2.与Java不同,c++不同new对象,对于无参的构造函数声明对象时括号应该省略. 2 析构函数 1. 前加~,不能有参数,不能有返回 ...

  7. 【C++基础】类的组合

    所谓类的组合是指:类中的成员数据是还有一个类的对象或者是还有一个类的指针或引用.通过类的组合能够在已有的抽象的基础上实现更复杂的抽象. 比如: 1.按值组合 #include<iostream. ...

  8. Java编程的逻辑 (14) - 类的组合

    ​本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http: ...

  9. java 报错非法的前向引用

    今天在看<thinking in java>的时候,第四章提到了非法的前向引用,于是自己试了一下,书中的例子倒是一下就明白了,但是自己写的一个却怎么也不明白,于是上网问了一位前辈,终于明白 ...

随机推荐

  1. SOS.DLL在windbg里加载错误

    sos.dll/mscordacwks.dll 公共语言运行库(CLR)是执行托管代码的Microsoft.NET框架的核心引擎.简单地说,它通过在托管程序集中使用中间语言和元数据,JIT按需编译代码 ...

  2. [RN] React Native 使用 FlatList 实现九宫格布局 GridList

    React Native 使用 FlatList 实现九宫格布局 先看图片演示实例: 本文以图片列表为例,实现九宫格布局! 主要有两种方法: 1)方法一: 利用FlatList的 numColumns ...

  3. [BZOJ1191]超级英雄Hero

    Description 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的 多少获得不同数目的奖品或奖金.主持人问题准备了若干道题目,只有当选手正确 ...

  4. 洛谷P1560 蜗牛的旅行

    题目 搜索,注意判断特殊情况,并且区分开什么时候转弯什么时候停止.然后转弯的时候更是要注意是否会进入障碍. #include <bits/stdc++.h> using namespace ...

  5. Java-Long类型精度丢失问题

    问题 今天碰到一个问题,后端需要返回给前端Long类型的id,前端收到的id会发生精度丢失. 测试代码:后端返回的值为344739147160346624 但是前端获取的值为: 解决办法 将返回的值转 ...

  6. mapreduce数据处理——统计排序

    接上篇https://www.cnblogs.com/sengzhao666/p/11850849.html 2.数据处理: ·统计最受欢迎的视频/文章的Top10访问次数 (id) ·按照地市统计最 ...

  7. ldap和phpldapadmin的安装部署

    LDAP 安装 一.安装LDAP 1. 安装包 yum install openssl-devel gcc libtool-ltdl-devel -y yum install openldap-ser ...

  8. 关于资源获取(请把https改为http)

    所有demo以及资源获取,请把https改为http.

  9. Java_jdbc 基础笔记之十 数据库连接 (ResultSetMetaData 类)

    ResultSetMetaData 类 调用ResultSet 的getMetaData()方法得到ResultSetMetaData 类对象: 可用于获取关于 ResultSet 对象中列的类型和属 ...

  10. vue---发送数据请求的一些列的问题

    使用vue做数据请求,首先考虑的是封装请求方法request.js import axios from 'axios' import Qs from 'qs' // 创建一个axios实例 const ...