类和对象(3)

文章目录

  • 1.回顾上节
  • 2. 拷贝构造
  • 3. 运算符重载(非常重要)
  • 4. 赋值运算符重载

1.回顾上节

在这里插入图片描述
默认成员函数:我们不写,编译器自动生成。我们不写,编译器不会自动生成
默认生成构造和析构:

  1. 对于内置类型不做处理
  2. 对于自定义类型会调用对应的构造/析构。

2. 拷贝构造

#include <iostream>
using namespace std;
class Date
{
public:Date(int year=1, int month=1, int day=1){_year = year;_month = month;_day = day;}//拷贝构造,函数名和类名相同//拷贝构造的参数为什么不能是传值?//C++自定义类型的成员在这个地方传值需要调用拷贝构造,无穷无尽//因此自定义类型必须调用拷贝构造,所以要用引用&。最好加constDate(const Date& d){_year = d._year;_month = d._month;_day = d._day;}void Print(){cout << _year << "年" << _month << "月" << _day << "日" << endl;}
private:int _year;int _month;int _day;
};
int main()
{Date d1;Date d2(d1);d1.Print();d2.Print();//这时改变_year等会改变d1return 0;
}

在这里插入图片描述
以下为不调用拷贝构造时,会默认生成拷贝构造
在这里插入图片描述
内置类型会处理,因此日期类不需要自己去写拷贝构造
自定义类型会去调用他的拷贝构造

Stack st1;Stack st2(st1);//栈中保持后进先出,后定义的先析构。//st1变成野指针。

指向同一块空间的问题:

  1. 插入删除数据会互相影响
  2. 析构两次,程序崩溃。
    默认的拷贝:
    浅拷贝/值拷贝
    **深拷贝:**让各自有各自独立的空间,开另外的空间,把值拷贝下来。
    更深入层次的拷贝
typedef int DataType;
class Stack
{
public:Stack(size_t capacity = 10){_array = (DataType*)malloc(capacity * sizeof(DataType));if (nullptr == _array){perror("malloc申请空间失败");exit(-1);}_size = 0;_capacity = capacity;}
void Push(const DataType& data)
{_array[_size] = data;_size++;
}
Stack(const Stack& st)//深拷贝
{_array = (DataType*)malloc(sizeof(DataType) * st._capacity);if (nullptr == _array){perror("malloc申请空间失败");exit(-1);//直接终止程序}//拷贝数组空间上的值memcpy(_array, st._array, sizeof(DataType) * st._size);_size = st._size;_capacity = st._capacity;
}
~Stack()
{if (_array){free(_array);_array = nullptr;_capacity = 0;_size = 0;}
}
private:DataType *_array;size_t _size;size_t _capacity;};
int main()
{Stack st1;st1.Push(1);st1.Push(2);st1.Push(3);st1.Push(4);Stack st2(st1);//栈中保持后进先出,后定义的先析构。//没有写拷贝构造,编译器自动生成了一个return 0;
}

什么情况下需要写拷贝构造呢?
不能用指针来衡量,如果自己实现了析构函数释放了空间,就需要实现拷贝构造。

  1. 对于内置类型完成浅拷贝/值拷贝–按byte一个个拷贝
  2. 自定义类型,去调用这个成员拷贝构造/赋值重载
    2种大方向的特性
typedef int DataType;
class Stack
{
public:Stack(size_t capacity = 10){_array = (DataType*)malloc(capacity * sizeof(DataType));if (nullptr == _array){perror("malloc申请空间失败");exit(-1);}_size = 0;_capacity = capacity;}
void Push(const DataType& data)
{_array[_size] = data;_size++;
}
Stack(const Stack& st)
{//拷贝构造对内置类型完成值拷贝或者浅拷贝。cout << "Stack(const Stack& st)" << endl;_array = (DataType*)malloc(sizeof(DataType) * st._capacity);if (nullptr == _array){perror("malloc申请空间失败");exit(-1);//直接终止程序}//拷贝数组空间上的值memcpy(_array, st._array, sizeof(DataType) * st._size);_size = st._size;_capacity = st._capacity;
}
~Stack()
{if (_array){free(_array);_array = nullptr;_capacity = 0;_size = 0;}
}private:DataType *_array;size_t _size;size_t _capacity;};
//对于自定义类型,不需要写拷贝构造和构造。不写编译器会自动生成构造函数,构造函数符合我们的需求
class MyQueue
{
public://默认生成构造和析构//默认生成拷贝构造
private:Stack _pushST;Stack _popST;int _size = 0;//缺省值,用缺省值处理
};
int main()
{Stack st1;st1.Push(1);st1.Push(2);st1.Push(3);st1.Push(4);Stack st2(st1);//栈中保持后进先出,后定义的先析构。//没有写拷贝构造,编译器自动生成了一个MyQueue q1;//调用了拷贝构造MyQueue q2(q1);return 0;
}

在这里插入图片描述
那些场景存在拷贝构造
Date d2(d1);
Date d3=d1;//拷贝构造
传返回值的过程中能用引用就用引用,减少拷贝。除非就是想让他自己调用拷贝构造,拷贝一份独立的出来
参数基本都可以用引用,返回值不一定。局部对象不能用引用。

Date Test(Date d)
{Date temp(d);return temp;
}

3. 运算符重载(非常重要)

为了增强程序的可读性,是具有特殊函数名的函数,也有其返回值类型函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似

  • 比较日期大小
    内置类型可以比较大小,自定义类型不可以
  • 运算符重载和函数重载无关:
    函数重载是支持参数名相同,参数不同的函数,随时可以用
    运算符重载:自定义类型对象可以使用运算符。
    两个地方都用了重载,但两个地方没有关联
  • 运算符重载:实现一个函数。新增一个关键字operator加操作符有参数有返回值
  • 参数和返回值根据运算符确定。有的有返回值有的没有。
  • 运算符有几个操作数就有几个参数
class Date
{
public:Date(int year=1, int month=1, int day=1){_year = year;_month = month;_day = day;}//private:int _year;int _month;int _day;};bool operator==(const Date& d1, const Date& d2)//运算符重载可以实现在全局。
{return d1._year == d2._year&& d1._month == d2._month&& d1._day == d2._day;
}
int main()
{Date d1(2023, 9, 14);Date d2(2023, 9, 14);cout<<operator==(d1, d2)<<endl;cout <<( d1 == d2) << endl;//全局函数,转换成调用这个函数operator==(d1,d2);和上一行一样//运算符优先级<<高于==return 0;
}

在这里插入图片描述
当放成私有时

class Date
{
public:Date(int year=1, int month=1, int day=1){_year = year;_month = month;_day = day;}
private:int _year;int _month;int _day;};

直接把函数放在类里面。类外面受到访问限定符的限制,放到类里面就解决问题了。但是会报错
在这里插入图片描述
其中还有隐藏的参数(2个):this
成员函数调用的方式也不同了。

//d1==d2转换为d1.operator==(d2)
bool operator==(const Date& d){//this:d1;d:d2return this->_year == d._year&& _month == d._month&& _day == d._day;}
cout<<d1.operator==(d2)<<endl;cout <<( d1 == d2) << endl;//成员函数转换成调用这个函数d1.operator==(d2);和上一行一样

在这里插入图片描述
运算符重载

  1. 函数名:operator+运算符或操作符
  2. 返回值类型/参数:根据需求调用
  3. 不能乱接其他符号创造一个新的操作符,如:operator@
  4. 必须有一个类类型参数**(自定义类型)**
  5. 不能对内置类型重载,其含义不能改变。如内置类型的整型——,不能改变其含义
  6. 作为类成员函数重载时,其形参看起来比操作数目少1,因为成员函数的第一个参数为隐藏的this
  7. .* :: sizeof ?:(三目运算符) .(成员访问) 注意以上5个运算符不能重载,这个经常在笔试选择题中出现
//b1<b2小测
bool operator<(const Date& d)
{if (_year < d._year){return true;}else if (_year == d._year && _month < d._month){return true;}else if (_year == d._year && _month == d._month && _day < d._day){return true;}else{return false;}
}

//b1<=b2复用,根据上面有<有=

bool operator<=(const Date& d)
{return *this < d || *this == d;
}

//b1>b2,取反。

bool operator>(const Date& d)
{return !(*this <= d) ;
}

4. 赋值运算符重载

d1=d2;//是一种拷贝

//d1=d2
void operator=(const Date& d)//不用引用不会无穷递归,但会白白走一次拷贝构造,所以最好加上引用
{_year = d._year;_month = d._month;_day = d._day;
}
d3=d2=d1;//编译不通过d1赋值给d2,d2的返回值传给d3

连续赋值,从右往左赋值。i=j=k; k赋值给j,返回j

Date& operator=(const Date& d)
{_year = d._year;_month = d._month;_day = d._day;//*this是d1return *this;//出了作用域还在,应该加引用。//返回值是为了支持连续赋值,保持运算符的特性。
}

d1=d1
自己给自己赋值,可以加一个判断

Date& operator=(const Date& d)//引用
{if(this!=&d)//取地址,this是左操作数的地址,d是右操作数的别名,地址相同则不用自己给自己赋值{_year = d._year;_month = d._month;_day = d._day;}return *this;
}

+=支持连续赋值,只要支持连续赋值就都有返回值。。
前置++,d1.operator();
后置++,d2.operator(int);
int仅仅是为了占位,和牵制重载区分

//++d1;
Date& Date::operator++()
{Date tmp(*this);*this+=1;return tmp;
}
//d1++
Date Date::operator++(int)
{Date tmp(*this);*this+=1;return tmp;
}

对于内置类型,前置和后置++没有区别
对于自定义类型,**前置++**效率高,后置++还要拷贝

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/132864.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PMP-项目规划过程组的重要性

一、什么是项目规划过程组 规划过程组包括明确项目全部范围、定义和优化目标&#xff0c;并为实现目标制定行动方案的一组过程。规划过程组中的过程制定项目管理计划的组成部分&#xff0c;以及用于执行项目的项目文件。取决于项目本身的性质&#xff0c;可能需要通过多轮反馈来…

使用阿里PAI DSW部署Stable Diffusion WebUI

进入到网址https://pai.console.aliyun.com/里边。 点击创建实例。 把实例名称填写好&#xff0c;选择GPU规格&#xff0c;然后选择实例名称是ecs.gn6v-c8g1.2xlarge。 选择stable-diffusion-webui-env:pytorch1.13-gpu-py310-cu117-ubuntu22.04&#xff0c;然后点击下一步。…

Python+requests编写的自动化测试项目

框架产生目的&#xff1a;公司走的是敏捷开发模式&#xff0c;编写这种框架是为了能够满足当前这种发展模式&#xff0c;用于前后端联调之前&#xff08;后端开发完接口&#xff0c;前端还没有将业务处理完毕的时候&#xff09;以及日后回归阶段&#xff0c;方便为自己腾出学(m…

C++之保存编译全部中间文件(二百一十五)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

【持续记录】深度学习环境配置

1080面对Transformer连勉强也算不上了&#xff0c;还是要去用小组公用的卡 完整记一个环境配置&#xff0c;方便后面自用✍️ nvidia-smi查看GPU信息 ** CUDA版本12.2 conda -V查询conda版本 22.9.0 新建conda环境 准备装python3.8 conda create --name caiman python3.8.2激…

04-JVM对象创建深度剖析

上一篇&#xff1a;03-JVM内存模型剖析与优化 对象创建的主要流程: 1.类加载检查 虚拟机遇到一条new指令时&#xff0c;首先将去检查这个指令的参数是否能在常量池中定位到一个类的符号引用&#xff0c;并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有…

ppt录制在哪?实用技巧分享!

在现代演示和培训中&#xff0c;PPT演示已经成为越来越流行的一种交流方式。而录制ppt也成为了很多商务演讲、教学讲解、产品演示等场合的必备技能。本文将为您介绍两种常见的ppt录制方式&#xff0c;帮助您轻松录制ppt演示的过程。 ppt录制在哪&#xff1f; ppt是一款流行的演…

【扩散模型】4、Improved DDPM | 引入可学习方差和余弦加噪机制来提升 DDPM

文章目录 一、背景二、Improved DDPM——提升 Log-likelihood2.1 可学习的方差2.2 改进 noise schedule2.3 降低梯度噪声 三、效果 论文&#xff1a;Improved Denoising Diffusion Probabilistic Models 代码&#xff1a;https://link.zhihu.com/?targethttps%3A//github.com…

【ELK】日志分析系统概述及部署(ELFK部署实验)

目录 一、ELK概述 1、ELK是什么&#xff1f; 2、ELK的组成部分 2.1 ElasticSearch &#xff08;1&#xff09;分片和副本 &#xff08;2&#xff09;es和传统数据库的区别 2.2 Kiabana 2.3 Logstash &#xff08;1&#xff09;Log Stash主要组件 2.4 可添加的其它组件…

YOLO DNF辅助教程完结

课程完结&#xff01;撒花、撒花、撒花 课程完结&#xff01;撒花、撒花、撒花 课程完结&#xff01;撒花、撒花、撒花 ​呕心沥血三个月&#xff0c;《利用人工智能做DNF游戏辅助》系列实战课程已完结&#xff0c;技术路线贯穿串口通信、目标检测、opencv特征匹配等前沿技术…

黑马JVM总结(八)

&#xff08;1&#xff09;StringTable面试题 1.8 1.6时 &#xff08;2&#xff09;StringTable的位置 jvm1.6时StringTable是常量池的一部分&#xff0c;它随着常量池存储在永久代当中&#xff0c;在1.7、1.8中从永久代变成了堆中&#xff0c;为什么做这个更改呢&#xff1f…

基于安卓Java试题库在线考试系统uniapp 微信小程序

本文首先分析了题库app应用程序的需求&#xff0c;从系统开发环境、系统目标、设计流程、功能设计等几个方面对系统进行了系统设计。开发出本题库app&#xff0c;主要实现了学生、教师、测试卷、试题、考试等。总体设计主要包括系统功能设计、该系统里充分综合应用Mysql数据库、…

金九银十,给大家一点面试方面的建议

好久不见&#xff0c;甚是想念。这段时间没有更新什么文章&#xff0c;其实是因为我跳了一波槽&#xff0c;出去面了一圈后&#xff0c;也顺利拿了不少架构岗位的offer。 正好马上要金九银十了&#xff0c;相信有不少小伙伴们估计也有跳槽涨薪的想法&#xff0c;那么就从我最近…

input修改checkbox复选框默认选中样式

问题描述&#xff1a; <input type"checkbox" /> input修改checkbox默认选中样式&#xff0c;直接设置选中后的样式不生效&#xff0c;需要先给复选框设置-webkit-appearance: none&#xff08;取消默认样式&#xff09;&#xff0c; 再设置样式才会生效。 …

面经pc端项目

创建项目 安装脚手架-----创建项目------选择自定义 sass基础语法 https://www.sass.hk/ sass语法有两个:sass(旧) scss(新) 1.scss语法 和less语法类似,支持嵌套,支持变量… scss: $变量名 less: @变量名 $color:orange; .box{width: 400px;height: 400px;borde…

Vue3高频面试题+八股文

Vue3.0中的Composition Api 开始之前 Compos:1 tion API可以说是ue3的最大特点&#xff0c;那么为什么要推出Compos1t1on Api,解决了什么问趣&#xff1f; 通常使用Vue2开发的项目&#xff0c;普遍会存在以下问题&#xff1a; 代码的可读性随着组件变大而变差每一种代码复用的…

勒索病毒最新变种.halo勒索病毒来袭,如何恢复受感染的数据?

摘要&#xff1a; .halo勒索病毒已成为数字世界中的威胁&#xff0c;通过高级加密技术将文件锁定&#xff0c;并要求支付赎金。本文91数据恢复将深入介绍.halo勒索病毒的工作原理&#xff0c;提供解锁被感染文件的方法&#xff0c;以及探讨如何有效预防这一威胁。如果您正在经…

数据库逻辑透明-架构真题(二十九)

&#xff08;2020年&#xff09;假设某计算机字长为32位&#xff0c;该计算机文件管理系统磁盘空间管理采用位示图&#xff08;bitmap&#xff09;记录磁盘的使用情况。若磁盘的容量为300GB&#xff0c;物理块大小为4MB&#xff0c;那么位示图的大小为&#xff08;&#xff09;…

JavaScript设计模式(五)——发布订阅模式、桥接模式、组合模式

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;正逐渐往全干发展 &#x1f4c3;个人状态&#xff1a; 研发工程师&#xff0c;现效力于中国工业软件事业 &#x1f680;人生格言&#xff1a; 积跬步…

项目:UDP聊天室

UDP UDP&#xff08;User Datagram Protocol&#xff09;是一种无连接、不可靠、面向数据报的传输协议。与TCP相比&#xff0c;UDP更加轻量级&#xff0c;不提供像TCP那样的可靠性和流控制机制&#xff0c;但具备较低的通信延迟和较少的开销。 UDP具有以下几个特点&#xff1…