【C++指南】一文总结C++类和对象【中】

🌟 各位看官好,我是egoist2023

🌍 种一棵树最好是十年前,其次是现在!

🚀 今天来学习C++类和对象的语法知识。注意:在本章节中,小编会以Date类举例

👍 如果觉得这篇文章有帮助,欢迎您一键三连,分享给更多人哦

目录

引入

运算符重载

赋值运算符重载

日期类实现

 日期计算

 日期判断

流插入<<  和  流提取>>

const成员函数和取地址运算符重载


引入

//C++中
Date d1(2024, 12, 24);
Date d2(2024, 12, 24);
d1 = d2;//C语言中
d1._year = d2._year;
d1._month = d2._month;
d1._day = d2_day;

在上面一段代码中,我们想把类类型的d2赋值给d1,如果在C语言版本中是需要把一个一个内置类型拷贝过去,未免过于麻烦。有什么方法能直接让类对象d2赋值给d1呢?在C++中提出了运算符重载的概念可以为类类型的对象指定新的含义。

运算符重载

  • C++规定类类型对象使用运算符时,必须调用对应运算符重载,若没有对应的运算符重载,则会编译报错
  • 运算符重载是一个函数,具有其返回类型和参数列表以及函数体,由operator和运算符共同构成
//在成员函数中
bool operator=(const Date* d)
    • 几元运算符对应有几个运算对象数量。如,一元运算符有一个运算对象,二元运算符有两个运算对象,且规定二元运算符左侧运算对象传给第一个参数,右侧传给第二个参数。(如果没有类类型形参则会报错,在全局函数中 int operator+(int x, int y) )
    • 重载<<和>>时,需要重载为全局函数(把ostream/istream放到第一个形参位置,第二个跟类类型对象)。  因为重载为成员函数,this指针默认为第一个形参位置,是左侧运算对象,调用时就变成了对象<<cout,用起来不习惯。
    ostream& operator<<(ostream& out, const Date& d);
    istream& operator>>(istream& in, Date& d);
    • 运算符重载不能连接语法中没有的符号,如operator@。 同时注意 .*   ::   sizeof   ?:  . 以上5个运算符不能重载。
    • 一个类是否需要重载哪些运算符,是看重载后是否有意义,比如Date类重载operator-就有意义(可以算差值多少天)。
    • 在C语言中有前置++和后置++,运算符重载函数名都是operator++,为了方便区分。C++规定,后置++重载时,增加⼀个int形参,跟前置++构成函数重载,方便区分。
    	Date& operator++();//前置++Date operator++(int);//后置++

    赋值运算符重载

    如下一行代码是否是赋值运算符重载呢?并不是,是拷贝构造。

    注意:拷贝构造是把一个已存在的对象拷贝初始化零一个要创建的对象;

    而赋值运算符重载是⽤于完成两个已经存在的对象直接的拷贝赋值。

    Date d1=d2//d1和d2都是类类型对象
    1. 赋值运算符重载 (跟拷贝构造类似)是⼀个运算符重载,必须重载为成员函数。赋值运算重载的参数建议 写成const 当前类类型引用 (减少拷贝)。
    返回值时也建议携程类类型引用(提高效率,支持返回值可提供连续赋值场景)。
    	Date& operator=(const Date& d){_year = d._year;_month = d._month;_day = d._day;return *this;}
    2.没有显式实现时,编译器会默认生成,对内置类型成员变量会浅拷贝(像Date类就可以不实现赋值赋值重载函数),对自定义类型成员变量会调用赋值重载函数(如 Stack ,_a指向了资源,默认生成的赋值重载函数不符合需求,需要 自己实现深拷贝 也对指向的资源也进行拷贝)。
    3.同样,对于MyQueue这样的类型内部主要是自定义Stack成员,编译器默认生成赋值运算符重载会调用Stack的赋值运算符重载。

    日期类实现

    构造函数Date(int year = 1, int month = 1, int day = 1)
    获取日期int GetMonthDay(int year, int month) const
    日期是否合法bool checkDate() const
    打印日期void Print()

    日期类构造函数不过多介绍。实现代码如下代码所示

    //构造函数
    Date::Date(int year, int month, int day)
    {_year = year;_month = month;_day = day;if (!checkDate()){cout << "日期非法:<" << *this << endl;}
    }

    已知某年某月,如何知道是何日呢? 由于该函数需要频繁调用,故设在类里面,类里默认为inline。

    	int GetMonthDay(int year, int month) const{static int MonthDayArray[13] = { -1,31,28,31,30,31,30,31,31,30,31,30,31 };if (month == 2 && ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0)){return 29;}return MonthDayArray[month];}

    如果输入的年月日 不符合实际上的年月日,需要一个函数检查是否合法。

    bool Date::checkDate() const
    {//非法返回falseif (_month < 1 || _month>12 || _day > GetMonthDay(_year, _month) || _day < 1)return false;elsereturn true;
    }
     日期计算
    日期+天数

        Date operator+(int day) const

    日期+=天数

     Date& operator+=(int day)

    日期-天数    Date& operator-=(int day)
    日期-=天数    Date operator-(int day) const
    日期-日期    int operator-(const Date& d) const
    前置/后置++    Date& operator++()   前置++
        Date operator++(int)  后置++
    前置/后置--    Date& operator--()     前置--
        Date operator--(int)    后置--

     

     那是第一种复用第二种,还是第二种复用第一种呢?我们来剖析下函数的具体行为是什么。

    第一种采用的是传值返回,为什么不传引用返回呢?因为返回的是tmp,tmp是建立在这个栈空间的,函数结束后栈会被销毁,因此要用传值返回。而C++规定类传值返回会调用拷贝构造,会有一定的消耗。

    第二种采用的是传引用返回,由于类类型d并不是在此栈空间的,传引用返回能减少拷贝,提高效率

    可以发现下图中第二种复用第一种会有5次消耗,而第一种复用第二种只存在2次消耗。

     


    同样日期-天数和日期-=天数也可以采用复用的逻辑。但其中还有借月的逻辑,需要细细来剖析。 

    Date& Date::operator-=(int day)
    {if (day < 0){*this += (-day);}_day -= day;while (_day <= 0){--_month;if (_month == 0){--_year;_month = 12;}_day += GetMonthDay(_year, _month);}return *this;
    }//优化
    Date Date::operator-(int day) const
    {Date tmp(*this);tmp -= day;return tmp;
    }

    由于实现了日期加减天数的函数,前置和后置的加减都可以采用复用的逻辑。 (这里就不过多赘述)

    //++d
    Date& Date::operator++()
    {//复用*this += 1;return *this;
    }//d++
    Date Date::operator++(int)
    {Date tmp(*this);*this += 1;return tmp;
    }//--d
    Date& Date::operator--()
    {*this -= 1;return *this;
    }//d--
    Date Date::operator--(int)
    {Date tmp(*this);*this -= 1;return tmp;
    }

    接下来我们来看看日期-日期的实现方法。

    当然,也可以通过一个变量来记录2024年4月5日到2025年3月7日的天数,同样采用复用的逻辑。

    int Date::operator-(const Date& d) const
    {int flag = 1;Date max = *this;Date min = d;if (*this < d){max = d;min = *this;flag = -1;}int count = 0;while (min != max){++min;++count;}return flag * count;
    }
     日期判断
    日期比较
    d1>d2bool operator>(const Date& d) const
    d1>=d2bool operator>=(const Date& d) const
    d1<d2bool operator<(const Date& d) const
    d1<=d2bool operator<=(const Date& d) const
    d1==d2bool operator==(const Date& d) const
    d1!=d2bool operator!=(const Date& d) const
    bool Date::operator>(const Date& d) const
    {if (_year > d._year){return true;}else if (_year == d._year && _month > d._month){return true;}else if (_year == d._year && _month == d._month){return _day > d._day;}return true;
    }bool Date::operator==(const Date& d) const
    {return _year == d._year&& _month == d._month&& _day == d._day;
    }

    在实现日期判断的时候,通常只要实现两个一个>和==的函数,其他判断只要复用这两个函数就可以实现。如要实现d1>=d2,则只要满足d1>d2或d1==d2;实现d1<d2,则满足!(d1>=d2)。

    流插入<<  和  流提取>>
    流插入<<ostream& operator<<(ostream& out, const Date& d)
    流提取>>istream& operator>>(istream& in, Date& d)

     前面我们说过需要把  流插入<<  和  流提取>> 设置为全局变量,否则不符合我们的使用习惯。

    需要注意的是,流插入时类类型d是可以+const,并不改变里面的值,而流提取不可以。

    如果我们输入的年月日是存在非法的情况的,因此需要判断。

    //流插入
    ostream& operator<<(ostream& out, const Date& d)
    {out << d._year << "年" << d._month << "月" << d._day << "日" << endl;return out;
    }//流提取
    istream& operator>>(istream& in, Date& d)
    {//in >> d._year >> d._month >> d._day;while (1){in >> d._year >> d._month >> d._day;if (d.checkDate()){break;}else {cout << "输入日期非法,请重新输入:<" << endl;}}return in;
    }

    const成员函数和取地址运算符重载

     const修饰的成员函数称之为const成员函数,放到成员函数参数列表的后面。
    void Print() const;
    const修饰的是成员函数隐含的this指针,表明在成员函数中不能对类的任何成员进行修改。
    Date* const this 加const修饰 const Date* const this
    取地址运算符重载分为普通取地址运算符重载和const取地址运算符重载,一般不需要去显式实现,使用一些特殊情景。如不想让别人取到当前类对象的地址,就可以自己实现,胡乱返回一个地址,调试的时候就会发现地址一直对不上。
    	Date* operator&(){//return this;return (Date*)0x0123753f;}const Date* operator&() const{//return this;return nullptr;}

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

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

    相关文章

    PgSql 操作技巧

    1、查询数据导出csv数据 \COPY (SELECT w.* from t_sys_warn w ) TO /home/cuadmin/warn_output.csv WITH CSV HEADER;2、导出sql Insert语句 pg_dump -U 用户名 -h 主机名 -p 端口号 -d 数据库名 --inserts -t 表名 > 导出文件.sqlpg_dump -U username -d dbname -t tabl…

    Unity ES3保存类的问题

    有以下一个物品类 public class Item_Base//基础物品 { public string ID; private Attribute_Data Item_attribute new(); } 当使用ES3保存这个类时&#xff0c;Item_attribute的数据不会被保存&#xff0c;因为它是私有private ES3保存类时&#xff0c;只会保存…

    react基本功

    useLayoutEffect useLayoutEffect 用于在浏览器重新绘制屏幕之前同步执行代码。它与 useEffect 相同,但执行时机不同。 主要特点 执行时机:useLayoutEffect 在 DOM 更新完成后同步执行,但在浏览器绘制之前。这使得它可以在浏览器渲染之前读取和修改 DOM,避免视觉上的闪烁…

    Spring Boot笔记(上)

    01 概要 Spring Boot 是 Java 领域最流行的 快速开发框架&#xff0c;专为简化 Spring 应用的初始搭建和开发而设计。 一、Spring Boot 解决了什么问题&#xff1f; 传统 Spring 痛点 • 繁琐的 XML 配置 • 需要手动管理依赖版本 • 部署依赖外部 Web 服务器&#xff08;如 …

    目标检测YOLO实战应用案例100讲-基于毫米波雷达的多目标检测 (续)

    目录 3.2 改进的CFAR目标检测算法 3.3 算法步骤描述 3.4 实验结果与分析 基于VGG16-Net的毫米波雷达目标检测算法 4.1 VGG16-Net网络模型 4.2 改进VGG16-Net网络的目标检测算法 4.3 算法步骤描述 4.4 实验结果与分析 知识拓展 基于毫米波雷达的多目标检测:使…

    gitsubtree怎么添加新的子仓库

    要使用 git subtree 添加一个新的子仓库&#xff0c;可以按照以下步骤操作&#xff1a; 1. 添加子仓库 使用 git subtree add 命令将子仓库的内容添加到主仓库的指定目录中。命令格式如下&#xff1a; git subtree add --prefix<子目录路径> <子仓库地址> <子…

    文本转语音-音画适时推送rtsp并播放

    文本语音 rtsp适时播放叫号系统的底层逻辑 发布Linux, unix socket 和window win32做为音频源的 python10下的(ffmpeg version 7.1) 可运行版本. 这两天在弄这个&#xff0c;前2篇是通过虚拟声卡&#xff0c;达到了最简单的一个逻辑&#xff0c;播放文本就从声卡发声&#xff0…

    从0开始的操作系统手搓教程33:挂载我们的文件系统

    目录 代码实现 添加到初始化上 上电看现象 挂载分区可能是一些朋友不理解的——实际上挂载就是将我们的文件系统封装好了的设备&#xff08;硬盘啊&#xff0c;SD卡啊&#xff0c;U盘啊等等&#xff09;&#xff0c;挂到我们的默认分区路径下。这样我们就能访问到了&#xff…

    【图片批量转换合并PDF】多个文件夹的图片以文件夹为单位批量合并成一个PDF,基于wpf的实现方案

    项目背景: 多个图片分布在不同文件夹,如何以文件夹为单位批量合并成一个PDF,还要保证文件夹里面图片大小和顺序 实现功能: 1、单张图片的转换PDF:一张图临时转一下 2、多张图片转换成PDF:多张图单独转成PDF 3、多级目录多张图转换成PDF:多级目录多张图单独转成多个PDF…

    如何用Kimi生成PPT?秒出PPT更高效!

    做PPT是不是总是让你头疼&#xff1f;&#x1f629; 快速制作出专业的PPT&#xff0c;今天我们要推荐两款超级好用的AI工具——Kimi 和 秒出PPT&#xff01;我们来看看哪一款更适合你吧&#xff01;&#x1f680; &#x1f947; Kimi&#xff1a;让PPT制作更轻松 Kimi的生成效…

    从 MongoDB 到 TDengine,沃太能源实现 18 倍写入性能提升

    导读 沃太能源是国内领先储能设备生产厂商&#xff0c;数十万储能终端遍布世界各地。此前使用 MongoDB 存储时序数据&#xff0c;但随着设备测点增加&#xff0c;MongoDB 在存储效率、写入性能、查询性能等方面暴露出短板。经过对比&#xff0c;沃太能源选择了专业时序数据库 …

    数据库基本建表操作

    1.登录数据库并创建数据库db_ck 创建完成后使用到我们创建的数据库。 2.创建表t_hero 根据hero属性包括&#xff08;id&#xff0c;name&#xff0c;nickname&#xff0c;age&#xff0c;gender&#xff0c;address&#xff0c;weapon&#xff0c;types&#xff09; 创建完…

    OkHttp 之任务调度模块源码分析

    一、引言 在现代网络应用开发中&#xff0c;高效的任务调度机制对于提升系统性能和用户体验至关重要。OkHttp 作为一款广泛使用的高性能 HTTP 客户端库&#xff0c;其任务调度模块在处理网络请求的并发、排队和执行等方面发挥着关键作用。本文将深入 OkHttp 源码&#xff0c;详…

    复现无人机的项目,项目名称为Evidential Detection and Tracking Collaboration

    项目名称为Evidential Detection and Tracking Collaboration&#xff0c;主要用于强大的反无人机系统&#xff0c;涉及新问题、基准和算法研究。下面介绍项目的复现步骤&#xff1a; 安装环境&#xff1a;使用Anaconda创建并激活名为edtc的虚拟环境&#xff0c;Python版本为3…

    QwQ-32B 开源!本地部署+微调教程来了

    今天&#xff0c;通义千问开源了推理模型QwQ-32B QwQ-32B 在一系列基准测试中进行了评估&#xff0c;测试了数学推理、编程能力和通用能力。以下结果展示了 QwQ-32B 与其他领先模型的性能对比&#xff0c;包括 DeepSeek-R1-Distilled-Qwen-32B、DeepSeek-R1-Distilled-Llama-7…

    如何利用 Excel 表格实现精准文件批量重命名教程

    在处理大量文件时&#xff0c;有时需要根据特定规则对文件名进行调整。如果您的文件名和新名称之间存在一对多的关系&#xff0c;并且这种关系可以通过 Excel 表格来管理&#xff0c;那么使用“简鹿文件批量重命名”软件中的“匹配对应名称命名”功能将是一个高效的选择。接下来…

    开关模式电源转换器 EMI/EMC 的集成仿真

    介绍 在电力电子领域&#xff0c;电磁干扰 &#xff08;EMI&#xff09; 和电磁兼容性 &#xff08;EMC&#xff09; 问题可以决定设计的成败。开关模式电源转换器虽然高效且紧凑&#xff0c;但却是电磁噪声的常见来源&#xff0c;可能会对附近的组件和系统造成严重破坏。随着…

    Android 蓝牙工具类封装:支持经典蓝牙与 BLE,兼容高版本权限

    为了优化经典蓝牙&#xff08;Classic Bluetooth&#xff09;和低功耗蓝牙&#xff08;Bluetooth Low Energy, BLE&#xff09;的操作&#xff0c;我们可以将功能封装到一个工具类中&#xff0c;支持扫描、连接、通信&#xff0c;并兼容高版本 Android 的动态权限申请。以下是完…

    STM32 CAN模块原理与应用详解

    目录 概述 一、CAN模块核心原理 1. CAN协议基础 2. STM32 CAN控制器结构 3. 波特率配置 二、CAN模块配置步骤&#xff08;基于HAL库&#xff09; 1. 初始化CAN外设 2. 配置过滤器 3. 启动CAN通信 三、数据收发实现 1. 发送数据帧 2. 接收数据帧&#xff08;中断方式…

    PostgreSQL_安装部署

    一、Windows系统下安装 1.下载安装包 登录PostgreSQL: Downloads官网&#xff1a; 选择14.12版本&#xff0c;点击下载&#xff1a; 2.安装PostgrSQL14.12 双击exe安装包程序&#xff0c;准备安装&#xff1a; 选择安装路径&#xff1a; 选择想安装的工具&#xff1a; 选择数…