【C++ 语言】面向对象 ( 函数重载 | 运算符重载 | 运算符重载两种定义方式 | 拷贝构造方法 | RVO 优化 | NRVO 优化 )(二)

简介: 【C++ 语言】面向对象 ( 函数重载 | 运算符重载 | 运算符重载两种定义方式 | 拷贝构造方法 | RVO 优化 | NRVO 优化 )(二)

拷贝构造方法


分析下面方法中的栈内存 ;


//运算符重载 , "+" 号运算符进行重载 , 
//  其作用是让两个 Operator 的 number 成员变量相加
//  运算符重载的本质是按照一定格式定义一个方法
//  这个定义的方法中包含运算符 , 除运算符之外的其它符号可以省略简写
public:
  Operator operator+(const Operator& o1) {
  //+ 运算符的作用是 两个 Operator 对象, 进行操作得到第三个 Operator 对象
  //第三个 Operator 对象的 number 变量 , 是前两个 Operator 对象之和
  Operator o2;
  o2.number = this->number + o1.number;
  //o2 对象是存在栈内存中 , 返回 o2 操作出现了拷贝操作
  //o2 会调用拷贝构造方法 , 拷贝到一个临时对象中 
  //如果返回值有接收的对象 , 那么又调用拷贝构造方法 , 
  //  将这个临时对象又会被拷贝给接收对象
  return o2;
  }



分析上述方法中的栈内存对象 , 在运算符重载的方法中 , 涉及到了栈内存对象的生命周期问题 , Operator o2; 语句在方法的栈内存中创建了一个 Operator 对象 o2 , 该对象的生命周期只限于整个方法体 , 方法执行完毕后 , 栈内存中的该对象就要被释放掉 ;


但是 该对象需要被返回 , 并且在方法调用完成之后 , 还要 赋值给另外一个对象 , 现在讨论一下其中的运行机制 ;


从 return o2; 开始分析 , 返回 o2 对象 , 系统会将栈内存中的 o2 对象 拷贝到一个临时对象中 , 这里调用了一次拷贝构造方法 ; 然后将临时对象又赋值给了返回值接收的对象 , 此处又调用了一次拷贝构造方法 ; 整个操作在理论上调用了两次拷贝构造方法 ;


拷贝构造方法实现 , 拷贝构造方法与构造方法的区别是 , 其需要传入一个引用类型 ( 类名& 变量名 ) 的参数 , 如下示例中实现了默认的构造方法 , 同时实现了拷贝构造方法 , 在发生该对象的拷贝操作时 , 会调用该方法 , 打印相应的数据 , 这样我们就能测试追踪对象的拷贝操作了 ;


/

/默认的构造方法
  Operator() {}
  //拷贝构造方法, 每次拷贝都会调用该构造方法
  //  以此来验证栈内存中 返回 栈内存中的对象 , 
  //  将栈内存对象拷贝到临时对象中
  //  在方法调用处 , 又将临时对象拷贝给了接收返回值的对象
  Operator(Operator& o) {
  this->number = o.number;
  cout << "Operator 对象执行拷贝操作" << endl;
  }


下面运行如下重载云算符调用的方法 , 分析其构造方法调用次数 , 下面是要运行的代码 :


//+ 是在 Operator 类中自定义的运算符重载 
  //其作用是返回一个对象 , 其number成员变量值是 o1 和 o2 中number成员变量之和
  Operator o4 = o1 + o2;
  //打印 o3 中的 number 变量值
  cout << "内部定义的运算符重载简化写法结果 : " << o4.number << endl;


运行结果如下 , 此处发现拷贝构造方法只调用了一次 , 理论上其应该调用两次 , 这就涉及到了编译器的 RVO 优化 ;


Operator 对象执行拷贝操作
内部定义的运算符重载完整写法结果 : 90



编译器优化 ( RVO 优化 | NRVO 优化 )


理论上拷贝构造方法是要执行两次的 , 在 operator+ 方法中 , 第一次将 o2 对象拷贝给临时对象 , 第二次将 临时对象拷贝给接收 operator+ 方法返回值的对象 ;


但是在 Visual Studio 中编译后执行结果 只拷贝了一次, 拷贝构造函数只调用了一次, 这是由于编译器优化的原因 ;


Windows 上 Visual Studio 的 C++ 编译器是 cl.exe

MAC 上 Xcode 的 C++ 编译器是 GNU g++

rvo 优化 , 在 VS 中, cl 编译器在 debug 模式下,会执行 rvo (return value) 优化 , 减少了1次拷贝和析构的操作 ;


nrvo 优化 , 在 release 模式下 , 会执行 nrvo 优化 , 会进行 0 次拷贝 , 减少了 2 次拷贝和析构的操作 , 其优化方式是改写方法 , 直接将接收对象放入参数 , 在方法中就将返回对象赋值给接收对象了 ;




完整代码示例


操作符重载类代码 :


#pragma once
using namespace std;
class Operator {
public : 
  int number;
  //默认的构造方法
  Operator() {}
  //拷贝构造方法, 每次拷贝都会调用该构造方法
  //  以此来验证栈内存中 返回 栈内存中的对象 , 
  //  将栈内存对象拷贝到临时对象中
  //  在方法调用处 , 又将临时对象拷贝给了接收返回值的对象
  Operator(Operator& o) {
  this->number = o.number;
  cout << "Operator 对象执行拷贝操作" << endl;
  }
  //这里针对拷贝操作进行说明 : rvo 优化 , nrvo 优化
  //理论上 拷贝 构造方法 是要执行两次的 , 在 operator+ 方法中 , 
  //  第一次将 o2 对象拷贝给临时对象
  //  第二次将 临时对象拷贝给接收 operator+ 方法返回值的对象
  //但是在 Visual Studio 中编译后执行结果只拷贝了一次, 这是由于编译器优化的原因.
  //  Windows 上 Visual Studio 的 C++ 编译器是 cl.exe
  //  MAC 上 Xcode 的 C++ 编译器是 GNU g++
  //在 VS 中, cl 编译器在 debug 模式下,会执行 rvo (return value) 优化
  //  rvo 优化 , 减少了1次拷贝和析构的操作
  //在 release 模式下 , 会执行 nrvo 优化
  //  nrvo 优化 , 会进行 0 次拷贝 , 减少了 2 次拷贝和析构的操作
  //  其优化方式是改写方法 , 直接将接收对象放入参数 , 在方法中就将返回对象赋值给接收对象了
  //这两种优化都是编译器针对返回值进行的优化
//类内部定义云算符重载
//运算符重载 , "+" 号运算符进行重载 , 
//  其作用是让两个 Operator 的 number 成员变量相加
//  运算符重载的本质是按照一定格式定义一个方法
//  这个定义的方法中包含运算符 , 除运算符之外的其它符号可以省略简写
public:
  Operator operator+(const Operator& o1) {
  //+ 运算符的作用是 两个 Operator 对象, 进行操作得到第三个 Operator 对象
  //第三个 Operator 对象的 number 变量 , 是前两个 Operator 对象之和
  Operator o2;
  o2.number = this->number + o1.number;
  //o2 对象是存在栈内存中 , 返回 o2 操作出现了拷贝操作
  //o2 会调用拷贝构造方法 , 拷贝到一个临时对象中 
  //如果返回值有接收的对象 , 那么又调用拷贝构造方法 , 
  //  将这个临时对象又会被拷贝给接收对象
  return o2;
  }
};
//类外部定义云算符重载
//  使用该重载云算符时 , 将两个对象相乘 , 获得的第三个对象 , 
//  该对象的 number 成员变量值 , 是 前两个对象的 number 对象的乘积 
Operator operator*(const Operator& o1, const Operator& o2) {
  //+ 运算符的作用是 两个 Operator 对象, 进行操作得到第三个 Operator 对象
  //第三个 Operator 对象的 number 变量 , 是前两个 Operator 对象之和
  Operator o3;
  o3.number = o1.number * o2.number;
  //o2 对象是存在栈内存中 , 返回 o2 操作出现了拷贝操作
  //o2 会调用拷贝构造方法 , 拷贝到一个临时对象中 
  //如果返回值有接收的对象 , 那么又调用拷贝构造方法 , 
  //  将这个临时对象又会被拷贝给接收对象
  return o3;
}





main 函数代码 :


// 003_Object_Oriented.cpp: 定义应用程序的入口点。
//
#include "003_Object_Oriented.h"
//引用 Student 类声明的头文件
#include "Student.h"
#include "Instance.h"
#include "Operator.h"
using namespace std;
void OOTest() {
  //在方法中直接声明 Student 对象, student 对象处于栈内存中 , 
  //其作用域仅限于 OOTest 函数 , 方法执行完就会清理掉
  Student student(18, 1);
}
void OOTest(int i) {
  //在方法中直接声明 Student 对象, student 对象处于栈内存中 , 
  //其作用域仅限于 OOTest 函数 , 方法执行完就会清理掉
  Student student(18, 1);
}
//友元函数实现 , 在类的外部修改类中的私有成员变量 age
void changeAge(Student* student){
  student->age = 88;
}
int main()
{
  cout << "Hello Student" << endl;
  OOTest();
  //在上面的 OOTest() 方法中的栈内存中创建了 Student 对象
  //当 OOTest() 方法执行完毕后 , 就会释放掉 Student 对象
  //使用 new 创建对象 , 注意该对象在堆内存中创建的 
  //用完之后需要使用 delete 释放该对象
  Student* student = new Student(18, 1);
  //调用友元函数, 修改 student 对象类私有变量 age 的值
  changeAge(student);
  //调用 getAge() 常量函数获取 student 对象的 age 成员变量值
  //并将该值打印出来
  cout<< "age : " << student->getAge() <<endl;
  //释放使用 new 申请的堆内存中的内存
  delete student;
  //创建单例对象
  Instance* instance = Instance::getInstance();
  //打印单例对象中的变量值
  cout << "单例 instance->number : " << instance->number << endl;
  //释放单例类
  delete instance;
  //运算符重载
  //注意这里的 Operator 对象 o1 和 o2 都在栈内存中
  Operator o1;
  o1.number = 80;
  Operator o2;
  o2.number = 10;
  //运算符重载完整写法
  //这是运算符重载的完整写法 , 
  //其中的 .operator 和之后的 () 可以省略变成下面的简化写法
  Operator o3 = o1.operator+(o2);
  //打印 o3 中的 number 变量值
  cout << "内部定义的运算符重载完整写法结果 : " << o3.number << endl;
  //运算符重载简化写法
  //+ 是在 Operator 类中自定义的运算符重载 
  //其作用是返回一个对象 , 其number成员变量值是 o1 和 o2 中number成员变量之和
  Operator o4 = o1 + o2;
  //打印 o3 中的 number 变量值
  cout << "内部定义的运算符重载简化写法结果 : " << o4.number << endl;
  //这里对栈内存说明一下
  //在运算符重载实现的方法中 , 创建了 Operator 对象, 
  //这个对象在方法返回时先拷贝给了一个临时对象
  //这个临时对象是一个不可见的匿名对象 , 对外透明的
  //返回该临时对象后 , 发现有 Operator o3 变量接收该对象
  //再次将临时对象拷贝给 o3 对象 
  //测试类外部的运算符重载
  //运算符重载完整写法
  //这是运算符重载的完整写法 , 
  //其中的 .operator 和之后的 () 可以省略变成下面的简化写法
  Operator o5 = operator*(o1, o2);
  //打印 o5 中的 number 变量值
  cout << "外部定义的运算符重载完整写法结果 : " << o5.number << endl;
  //运算符重载简化写法
  //+ 是在 Operator 类中自定义的运算符重载 
  //其作用是返回一个对象 , 其number成员变量值是 o1 和 o2 中number成员变量之积
  Operator o6 = o1 * o2;
  //打印 o6 中的 number 变量值
  cout << "外部定义的运算符重载简化写法结果 : " << o6.number << endl;
  return 0;


运行结果 :


Hello Student
Student() 构造方法
~Student() 析构方法
Student() 构造方法
age : 88
~Student() 析构方法
单例 instance->number : 888
Operator 对象执行拷贝操作
内部定义的运算符重载完整写法结果 : 90
Operator 对象执行拷贝操作
内部定义的运算符重载简化写法结果 : 90
Operator 对象执行拷贝操作
外部定义的运算符重载完整写法结果 : 800
Operator 对象执行拷贝操作
外部定义的运算符重载简化写法结果 : 800


目录
相关文章
|
3月前
|
存储 C++
C++语言中指针变量int和取值操作ptr详细说明。
总结起来,在 C++ 中正确理解和运用 int 类型地址及其相关取值、设定等操纵至关重要且基础性强:定义 int 类型 pointer 需加星号;初始化 pointer 需配合 & 取址;读写 pointer 执向之处需配合 * 解引用操纵进行。
289 12
|
8月前
|
存储 负载均衡 算法
基于 C++ 语言的迪杰斯特拉算法在局域网计算机管理中的应用剖析
在局域网计算机管理中,迪杰斯特拉算法用于优化网络路径、分配资源和定位故障节点,确保高效稳定的网络环境。该算法通过计算最短路径,提升数据传输速率与稳定性,实现负载均衡并快速排除故障。C++代码示例展示了其在网络模拟中的应用,为企业信息化建设提供有力支持。
206 15
|
8月前
|
存储 算法 安全
企业员工数据泄露防范策略:基于 C++ 语言的布隆过滤器算法剖析[如何防止员工泄密]
企业运营过程中,防范员工泄密是信息安全领域的核心议题。员工泄密可能致使企业核心数据、商业机密等关键资产的流失,进而给企业造成严重损失。为应对这一挑战,借助恰当的数据结构与算法成为强化信息防护的有效路径。本文专注于 C++ 语言中的布隆过滤器算法,深入探究其在防范员工泄密场景中的应用。
150 8
|
8月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
424 6
|
10月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
189 19
|
10月前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
281 13
|
10月前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
207 5
|
10月前
|
存储 C++
【C++面向对象——输入输出流】处理二进制文件(头歌实践教学平台习题)【合集】
本任务要求使用C++读取二进制文件并在每行前添加行号后输出到控制台。主要内容包括: 1. **任务描述**:用二进制方式打开指定文件,为每一行添加行号并输出。 2. **相关知识**: - 流类库中常用的类及其成员函数(如`iostream`、`fstream`等)。 - 标准输入输出及格式控制(如`cin`、`cout`和`iomanip`中的格式化函数)。 - 文件的应用方法(文本文件和二进制文件的读写操作)。 3. **编程要求**:编写程序,通过命令行参数传递文件名,使用`getline`读取数据并用`cout`输出带行号的内容。 4. **实验步骤**:参考实验指
222 5
|
10月前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
146 5
|
9月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。