C++——日期类

前言:哈喽小伙伴们,在上一篇文章中我们对C++类与对象的前半段知识进行了简单的分享,其中比较重要的莫过于C++类的六个默认成员函数

所以这篇文章,我们通过实现一个完整的日期的操作,来对这些成员函数有一个更加深入的理解


目录

一.基本框架

二.日期的比较

三.日期的加减运算

 1.得到月的天数

2.日期的加运算

3.日期的减运算

4.日期的++--运算

5.日期减日期

6.日期的输入输出

7.存在的问题

总结


一.基本框架

根据我们过去实现项目的方法,我们需要将声明与定义分离,同时还要实现测试代码与源代码分离,所以我们需要三个文件:

随后进行类的创建,基本成员函数的实现,以及测试代码的创建等框架。 

随后进行框架的测试:


二.日期的比较

两个日期之间的比较方式有很多种:>、<、<=、>=、==、!=

这些就需要我们的赋值运算符构造函数出马了。

上篇文章我们已经知道的“>”的写法:


bool operator>(const Date& d)
{if (_year > d._year)return true;else if (_year == d._year){if (_month > d._month)return true;else if (_month == d._month)return _day > d._day;elsereturn false;}elsereturn false;
}

但是就这一个的写法,就已经是很多,很麻烦的一段代码了,难道像这样的代码我们一共要写6个吗???当然不需要,我们要知道,这些符号之间都有两两互补的关系

比如说,我们现在写出了“>”,那么“<=”不就是“>”取反吗

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

 我们建议把“==”给写出来,因为它比较容易写:

bool Date::operator==(const Date& d)
{return _year == d._year&& _month == d._month&& _day == d._day;
}

如此一来,“!=”的写法就会是:

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

现在我们同时拥有了“>”和“==”,那么将两者结合自然就得到了“>=”

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

这样是不是非常的简洁???其余符号的代码就不一一列举啦,详情请看最后的完整代码展示。


三.日期的加减运算

日期的加减是一个相对比较困难的运算,它不像数字的加减那样简单,因为不仅存在大小月的天数不一,而且每四年还会出现闰年的特殊情况,这样就会导致进位非常的麻烦。下面我们就来详细分享一下,如此复杂的日期运算,到底该怎么实现。 


 1.得到月的天数

首先很重要的一点就是我们要能够知道每个月都分别有多少天,同时还有2月这个特殊的月份,我们通过一个函数来实现:

int GetMonthdays(int year, int month)
{assert(month > 0 && month < 13);int Monthdays[12] = { 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 Monthdays[month - 1];
}

首先要做的就是assert断言,防止月份输入错误,其次因为闰年是斯四年一次,所以我们默认情况下都是平年,通过数组来记录,能够方便获取。最后就是进行闰年二月的判断,如果2月,我们就去判断一下是否是闰年。 


2.日期的加运算

我们之间搬出代码来讲:

Date& Date::operator+(int day)
{_day += day;while(_day > GetMonthdays(_year, _month)){_day -= GetMonthdays(_year, _month);_month++;if (_month == 13){_year++;_month = 1;}}return *this;
}

依旧是使用赋值运算符构造函数,我们直接让_day加上我们要加的天数,随后进行判断,如果相加之后的天数大于当月的天数,就让_day减去该月的天数,剩下的自然就是下个月的天数,同时月份+1,如果月份+1后是13,那就需要向年进一,同时月份回到1

之所以使用循环,是因为如果我要加100天,那向月的进位就不止1了,所以要循环往复的判断

下面我们进行测试:

#include"Date.h"
int main()
{Date d1(2024, 2, 1);Date d2 = d1 + 30;d2.Print();return 0;
}

结果如下: 

1+30 = 31,而2024年恰巧就是闰年所以2月有29天31 - 29 = 2,所以结果为2024/3/2。 

但是这样的写法看似完美,但实际上存在一个很大的错误,来看代码:

#include"Date.h"
int main()
{Date d1(2024, 2, 1);Date d2 = d1 + 30;d2.Print();d1.Print();return 0;
}

 我们是让d2对象去接收d1对象的日期加上20天后的日期,但实际上:

d1对象的日期也发生了改变

这个错误其实也是可以理解的,因为我们在函数中直接默认进行操作的就是d1的成员变量。而这样的运算,实际上是“+=”运算。

所以想要保证d1的成员变量不变,就必须创建一个临时变量来代替

Date Date::operator+(int day)
{Date tmp(*this);tmp._day += day;while (tmp._day > GetMonthdays(tmp._year, tmp._month)){tmp._day -= GetMonthdays(tmp._year, tmp._month);tmp._month++;if (tmp._month == 13){tmp._year++;tmp._month = 1;}}return tmp;
}

 创建临时变量,就用到了我们的拷贝构造函数使用tmp临时变量代替d1对象进行操作

值得注意的一点是,由于tmp是临时的变量,当这个函数结束时就不存在了所以其作为返回值时,返回类型不能是引用。 

再进行测试,结果如下:

 


3.日期的减运算

理解了加运算之后,减运算的写法相信小伙伴们都能够自己悟出来了。

唯一值得注意的是,日期没有0天

//日期减等运算
Date& Date::operator-=(int day)
{_day -= day;while (_day <= 0){_month--;if (_month == 0){_year--;_month = 12;}_day += GetMonthdays(_year, _month);}return *this;
}

这里我们先来实现一下“-=”运算,注意while循环的判断条件,因为_day不可能等于0

如果当月剩余的天数不够用,就需要去借用上个月的天数继续减。结果如下:

那么问题来了,博主为什么要先实现“-=”呢 ???

下面我们就来看看“-”运算的实现:

//日期减运算
Date Date::operator-(int day)
{Date tmp(*this);tmp -= day;return tmp;
}

怎么样,有没有很震惊,为了不改变d1对象,我们确实创建了临时变量tmp,但是我们大可不必去再写像上边那样的一大长串代码因为我们已经有“-=”运算了,所以我们直接让tmp去进行“-=”运算,就可以得到结果: 

而我们前边实现过的加运算同样可以借用“+=”运算来写

//日期加运算
Date Date::operator+(int day)
{Date tmp(*this);tmp += day;return tmp;
}

4.日期的++--运算

我们知道,“++”和“--”运算都有前置和后置两种方式,那么我们该怎么用构造函数去分别实现呢?

不管是前置还是后置,它们都会有++,那么我们使用赋值运算符重载函数函数名该怎么写?难道也是一前一后???

并不是,实际上是使用函数重载来区分它们

	//前置++运算Date& operator++();//后置++运算Date operator++(int);

对于后置++,给它一个int参数,但是该参数并不会使用,只是用作编译器的区分

那么两个函数又该怎么实现呢??? 

要注意的是,前置++是先加1,再给值,而后置++是先给值,再++,所以后者就需要一个临时变量,我们同样借用一下“+=”函数

//前置++运算
Date& Date::operator++()
{*this += 1;return *this;
}
//后置++运算
Date Date::operator++(int)
{Date tmp = *this;*this += 1;return tmp;
}

再来进行测试: 

 

如此便可以实现“++”的运算符重载。“--”与之类似,博主这里就不做展开讲解


5.日期减日期

上边我们讲的日期减运算,是用日期去减去明确的天数得到一个新的日期

那么现在如果想用一个日期减去另一个日期,计算两个日期之间有多少天,又该怎么搞呢???

这个事情看似复杂,实则代码写起来也挺简单,现在给大家一个思想:

先去比较两个日期谁,然后我让小的一直去++并计数,直到跟大的相等计数的结果不就是两者的相差天数吗???

//日期-日期
int Date::operator-(const Date& d)
{Date max = *this;Date min = d;int flag = 1;if (*this < d){flag = -1;max = d;min = *this;}int n = 0;while (min < max){min++;n++;}return n * flag;
}

先默认前一个值为较大值,后一个为较小值,然后去比较如果前一个实际上是较小值,则进行互换,同时创建一个flag如果是大-小,结果即为整数,反之则赋值为-1,结果为负数,测试如下:

 


6.日期的输入输出

我们前边讲述的日期,都是我们在创建对象时就给定的数据,输出时也是用的Print函数。而且我们知道,cin和cout是无法直接输入输出自定义类型的数据的

那现在我们就想先创建一个对象,然后通过cin和cout来输入输出数据,该如何实现呢???

首先我们要知道,cin是istream类型的对象,而cout是ostream类型的对象,那么我们就可以通过赋值运算符重载函数来重载“>>”和“<<”两个符号来实现

//日期输出
void Date::operator<<(ostream& out)
{out << _year << "年" << _month << "月" << _day << "日" << endl;
}

但是这样的写法存在问题

赋值运算符重载函数定义在类中作为成员函数时,其第一个参数就会是默认的隐藏的this参数,也就是d1,而cout则是第二个参数,这就导致我们调用函数时两个实参的顺序存在问题,如果将其改为d1<<cout,就能通过编译:

但是这显然不符合我们C++的使用规范,所以想要恢复顺序,就需要将此函数定义在类外,交换两个形参的位置

但是这个时候又出现了问题,因为该函数在类外,而类的成员变量是私有的,我们不能使用

 

又该如何解决这个问题呢?

这就需要用到关键字:friend通过friend,将类外函数在类内进行友元声明,就可以啦:

 

但是此时还有一个问题,在C++中cout是支持同时输出多个变量,但是我们定义的函数却不行:

 

这是因为按照从左到右的顺序,执行完cout<<d1之后,它们需要返回一个ostream类型的返回值来继续和d2一起作为参数去继续调用函数,所以需要将该函数的返回值类型替换为ostream并返回out

//日期输出
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;return in;
}

首先就是返回值类型和参数类型为istream&,其次要注意参数d不能用const修饰,因为就是要给它输入值

测试如下:


7.存在的问题

 到这里呢,日期类的所有基本功能已经全部实现了,但是任然存在一个问题:

我们不小心将2月的天数传了个40,这怎么能允许呢,2月最多也就29天,40天怎么可能呢?但是发现d2还是按部就班的进行了“+”运算,这就会出现很大的问题。所以我们需要进行传入检查。

因为在构造函数和输入函数中都需要进行检查,所以我们需要一个创建一个函数

//检查日期合法性
bool Date::CheckInvalid()
{if (_year <= 0 || _month < 1 || _month > 12|| _day < 1 || _day > GetMonthdays(_year, _month))return false;elsereturn true;
}

 分别判断年,月,日是否都合法

//初始化
Date::Date(int year, int month, int day)
{_year = year;_month = month;_day = day;if (!CheckInvalid()){cout << *this << "该日期非法" << endl;exit(-1);}
}

构造函数中使用,若非法直接结束程序

//日期输入
istream& operator>>(istream& in, Date& d)
{while(1){in >> d._year >> d._month >> d._day;if (!d.CheckInvalid())cout << "输入的日期非法,请重新输入:" << endl;elsebreak;}return in;
}

输入函数中使用,若非法则重新输入: 

 


总结

日期类的实现到这里就分享完啦,希望能够帮助小伙伴们更加深入的理解类的内部结构及其成员函数的操作实现。

喜欢博主文章的小伙伴记得一键三连哦,我们下期再见!

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

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

相关文章

RabbitMQ-高级篇

服务异步通信-高级篇 消息队列在使用过程中&#xff0c;面临着很多实际问题需要思考&#xff1a; 1.消息可靠性 消息从发送&#xff0c;到消费者接收&#xff0c;会经理多个过程&#xff1a; 其中的每一步都可能导致消息丢失&#xff0c;常见的丢失原因包括&#xff1a; 发送…

Android 高德地图切换图层

一、默认样式 Android 地图 SDK 提供了几种预置的地图图层&#xff0c;包括卫星图、白昼地图&#xff08;即最常见的黄白色地图&#xff09;、夜景地图、导航地图、路况图层。 findViewById<TextView>(R.id.normal).setOnClickListener {updateSelectedStatus(TYPE_NORMA…

Glide完全解读

一&#xff0c;概述 glide作为android流行的图片加载框架&#xff0c;笔者认为有必要对此完全解读。glide提供了三级缓存、生命周期Destroy后自动移除缓存、自动适配ImageView&#xff0c;以及提供了各种对图片修饰的操作&#xff0c;如剪裁等。本文通过最简单的使用&#xff…

Vue(二十):ElementUI 扩展实现表格组件的拖拽行

效果 源码 注意&#xff1a; 表格组件必须添加 row-key 属性&#xff0c;用来优化表格的渲染 <template><el-row :gutter"10"><el-col :span"12"><el-card class"card"><el-scrollbar><span>注意: 表格组件…

自动化测试再升级,大模型与软件测试相结合

近年来&#xff0c;软件行业一直在迅速发展&#xff0c;为了保证软件质量和提高效率&#xff0c;软件测试领域也在不断演进。如今&#xff0c;大模型技术的崛起为软件测试带来了前所未有的智能化浪潮。 软件测试一直是确保软件质量的关键环节&#xff0c;但传统的手动测试方法存…

编写交互式 Shell 脚本

在日常的系统管理和自动化任务中&#xff0c;使用 Shell 脚本可以为我们节省大量时间和精力。 文章将以输入 IP 为例&#xff0c;通过几个版本逐步完善一个案例。 原始需求 编写一个交互式的 Shell 脚本&#xff0c;运行时让用户可以输入IP地址&#xff0c;并且脚本会将输入…

国辰智企TMS定制化模块,实现智慧园区的全面管理

智慧园区综合管理系统是一种针对园区业务场景的高度定制化解决方案&#xff0c;通过选择性部署相应的模块&#xff0c;实现对园区各方面业务的全面管理。通常情况下&#xff0c;园区都需要有效地管理资产、确保安全&#xff0c;以及进行访客预约。这一全面性的系统通过各个模块…

windows 谷歌浏览器Chrome 怎么禁止更新

1.首先把任务管理器里的谷歌浏览器程序结束&#xff1a; &#xff08;鼠标在任务栏右击&#xff0c;出现任务管理器&#xff09; 2.windowr&#xff0c;输入services.msc 带有Google Update的服务&#xff0c;选择禁用。 3.windowr&#xff0c;输入taskschd.msc 任务计划程序…

二叉搜索树,力扣

目录 题目地址&#xff1a; 题目&#xff1a; 我们直接看题解吧&#xff1a; 解题分析&#xff1a; 解题思路&#xff1a; 代码实现&#xff1a; 代码补充说明&#xff1a; 代码实现(中序遍历)&#xff1a; 题目地址&#xff1a; 98. 验证二叉搜索树 - 力扣&#xff08;LeetCod…

delete、truncate和drop区别

一、从执行速度上来说 drop > truncate >> DELETE 二、从原理上讲 1、DELETE DELETE from TABLE_NAME where xxx1.1、DELETE属于数据库DML操作语言&#xff0c;只删除数据不删除表的结构&#xff0c;会走事务&#xff0c;执行时会触发trigger&#xff08; 触发器…

8. 字符串转换整数 (atoi)-LeetCode(Java)

8. 字符串转换整数 (atoi) 题目&#xff1a;8. 字符串转换整数 (atoi) 请你来实现一个 myAtoi(string s) 函数&#xff0c;使其能将字符串转换成一个 32 位有符号整数&#xff08;类似 C/C 中的 atoi 函数&#xff09;。 函数 myAtoi(string s) 的算法如下&#xff1a; 读入…

AI大语言模型学习笔记之三:协同深度学习的黑魔法 - GPU与Transformer模型

Transformer模型的崛起标志着人类在自然语言处理&#xff08;NLP&#xff09;和其他序列建模任务中取得了显著的突破性进展&#xff0c;而这一成就离不开GPU&#xff08;图形处理单元&#xff09;在深度学习中的高效率协同计算和处理。 Transformer模型是由Vaswani等人在2017年…

2024美赛预测算法 | 回归预测 | Matlab基于WOA-LSSVM鲸鱼算法优化最小二乘支持向量机的数据多输入单输出回归预测

2024美赛预测算法 | 回归预测 | Matlab基于WOA-LSSVM鲸鱼算法优化最小二乘支持向量机的数据多输入单输出回归预测 目录 2024美赛预测算法 | 回归预测 | Matlab基于WOA-LSSVM鲸鱼算法优化最小二乘支持向量机的数据多输入单输出回归预测预测效果基本介绍程序设计参考资料 预测效果…

Git 指令

Git 安装 操作 命令行 简介&#xff1a; Git 是一个开源的分布式版本控制系统&#xff0c;用于敏捷高效地处理任何或小或大的项目。 Git 是 Linus Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。 Git 与常用的版本控制工具 CVS, Subversion …

2024PMP考试新考纲-【业务环境领域】典型真题和很详细解析(2)

华研荟继续分享【业务环境Business Environment领域】在新考纲下的真题&#xff0c;帮助大家体会和理解新考纲下PMP的考试特点和如何应用所学的知识和常识&#xff08;经验&#xff09;来解题&#xff0c;并且举一反三&#xff0c;一次性3A通过2024年PMP考试。 2024年PMP考试新…

准确率90%+!大模型会话洞察平台来了

随着客户行为和需求加速改变&#xff0c;企业与客户在数字渠道沟通并交易的比重大幅提升。企业通过在线客服、社交媒体、短信、语音助手等数字化渠道与客户建立联系的方式&#xff0c;不仅拓宽了沟通途径&#xff0c;更显著提高了服务效率和质量。 与此同时&#xff0c;数字化…

POSIX(包含程序的可移植性) -- 详解

1. 什么是 POSIX 参考链接–知乎 POSIX 标准包含了进程管理、文件管理、网络通信、线程和同步、信号处理等方面的功能。 这些接口定义了函数、数据类型和常量等&#xff0c;为开发者提供了一个可移植的方法来与操作系统进行交互。 2. 谁遵守这个标准 遵守 POSIX 标准的主要是…

蛋氨酸市场调研:预计2029年将达到87亿美元

蛋氨酸&#xff0c;又名甲硫氨酸&#xff0c;化学名称为γ-甲硫基-a-氨基丁酸&#xff0c;是一种参与蛋白质合成的基本结构单位&#xff0c;是人体内八种必需氨基酸之一&#xff0c;同时也是重要的饲料添加剂。蛋氨酸主要用于家禽、猪的饲料添加剂以及药用等。对禽类来说&#…

adb脚本操作

用荣耀80手机测试 echo off setlocal enabledelayedexpansion adb shell am start com.android.settings timeout /t 2 /nobreak >nul adb shell input tap 500 1300 timeout /t 2 /nobreak >nul adb shell input tap 500 800 timeout /t 2 /nobreak >nul adb she…

ElasticSearch搜索与分析引擎-Linux离线环境安装教程

目录 一、下载安装包 网盘链接: 二、安装流程及遇到的问题和解决方案 &#xff08;1&#xff09;JDK安装 &#xff08;2&#xff09;Elasticsearch安装 &#xff08;3&#xff09;Kibana安装 ​&#xff08;4&#xff09;Ik分词器安装 三、启动过程中的问题 &#xff…