日期类实现(接类和对象联系一) - 实践

news/2025/10/24 14:24:58/文章来源:https://www.cnblogs.com/lxjshuju/p/19163218

我们要实现日期的加减、大小比较等功能。

一、声明

这里先创建一个头文件来进行定义声明:

Date(int year = 1900, int month = 1, int day = 1);
void Print();

创建Date对象时初始化日期(年、月、日)

void Print();作为普通成员函数,用于日期打印。

	int GetMonthDay(int year, int month){assert(month > 0 && month < 13);static int monthDayArray[13] = { -1, 31, 28, 31, 30, 31, 30,//0月为-1,没有
31, 31, 30, 31, 30, 31 };if (month == 2 && (year % 4 == 0 && year % 100 != 0) || (year% 400 == 0)){return 29;}else{return monthDayArray[month];}return monthDayArray[month];}bool CheckDate();

我们创建一个月份函数GetMonthDay,用于判断每个月固定的天数是多少,并且判断是否为润年。在这因为要频繁调用月份函数,所以建议使用静态数组减少开销。

CheckDate是用于判断,输入的日期是否有误。

//声明定义分离
bool operator<(const Date& d);
bool operator<=(const Date& d);
bool operator>(const Date& d);
bool operator>=(const Date& d);
bool operator==(const Date& d);
bool operator!=(const Date& d);
Date& operator+=(int day);
Date operator+(int day);
Date& operator-=(int day);
Date operator-(int day);
Date& operator++();
Date operator++(int);
int operator-(const Date& d);

上面这些就是我们要实现的功能,operator 作为专门用于重载运算符的关键字,我们在这给她先声明。

private:int _year;int _day;int _month;
};

这里就是声明我们的年月日了。

//友元函数声明后可以访问私有函数:
friend ostream& operator<<(ostream& out, const Date& d);
friend istream& operator>>(istream& in, Date& d);
ostream& operator<<(ostream& out, const Date& d);
istream& operator>>(istream& in, Date& d);

上面是流的重载方便我们进行日期输入和输出,friend 是友元函数的定义,使用友元函数可以访问私有成员函数。

值得注意的是,流的重写不能放在类中,否则会因为内置的隐藏参数this指针导致会以奇怪的方式实现原功能。

下面就是完整的头文件Date.h:

#pragma once
#include
using namespace std;
#include
class Date
{//友元函数声明后可以访问私有函数:friend ostream& operator<<(ostream& out, const Date& d);friend istream& operator>>(istream& in, Date& d);
public:Date(int year = 1900, int month = 1, int day = 1);void Print();//获取某一年某一天//GetMonthDay没有声明和定义分离,默认是inline函数,对于这种小而频繁调用的函数设为内联函数减少开销int GetMonthDay(int year, int month){assert(month > 0 && month < 13);static int monthDayArray[13] = { -1, 31, 28, 31, 30, 31, 30,//0月为-1,没有
31, 31, 30, 31, 30, 31 };//因为频繁调用月分,直接放入静态区节省开销//闰年判断if (month == 2 && (year % 4 == 0 && year % 100 != 0) || (year% 400 == 0)){return 29;}else{return monthDayArray[month];}return monthDayArray[month];}bool CheckDate();//判断是否输入错误日期//声明定义分离bool operator<(const Date& d);bool operator<=(const Date& d);bool operator>(const Date& d);bool operator>=(const Date& d);bool operator==(const Date& d);bool operator!=(const Date& d);Date& operator+=(int day);Date operator+(int day);Date& operator-=(int day);Date operator-(int day);Date& operator++();Date operator++(int);//这里有++d1和d1++两种形态,所以规定实现++重载的是前置++d1,后置++重载需要添加一个int形参,入d1.operator(0)int operator-(const Date& d);
private://在重载流要使用这三个得设为全局函数,或者增加三个公有函数//或者用另一种方法,友元元声明int _year;int _day;int _month;
};
//流重载
//流重载若放在成员函数里,会不可避免的出现左右操作数要按顺序对齐,导致原本cout<>(istream& in, Date& d);

二、实现

void Date::Print()
{cout << _year << "-" << _month << "-" << _day << endl;
}

函数打印,输出我们输入的日期。

bool Date::CheckDate()
{if (_month < 1 || _month > 12|| _day < 1 || _day > GetMonthDay(_year, _month))//不属于一到十二月,和1到月底日数{return false;}else{return true;}
}
Date::Date(int year, int month, int day)
{_year = year;_month = month;_day = day;if (!CheckDate()) {cout << "非法日期:" << endl;Print();}
}

实现我们的年份判断,当月份和天数不在GetMonthDay规定的对应月份所拥有的天数中和月份不在1-12月时,返回false,并且打印出错误日期。

bool Date:: 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) {if (_day < d._day) {return true;}}else {return false ;}}
}

我们开始实现小于比较,通过逐步对比年月日来进行判断。

bool Date:: operator <= (const Date& d) {return *this < d || *this == d;}bool Date:: operator >= (const Date& d) {return *this > d || *this == d;}
bool Date:: operator > (const Date& d) {//return *this < d || *this == 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);
}

这里可以使用复用简化代码来实现他们的功能,但要注意的是必须有一个运算符按照<符的方式重写,否则会调入无线调用的死循环里。

Date& Date::operator+=(int day)
{	if (day < 0) {return *this -= (-day);}_day += day;while (_day > GetMonthDay(_year, _month)){_day -= GetMonthDay(_year, _month);++_month;//进入下一个月if (_month == 13)//如果出现13月{++_year;//进入下一年,月份归为一月_month = 1;}}return *this;
}
Date Date::operator+(int day)
{Date tmp = *this;tmp += day;//函数复用,不用费那么多脑细胞再写一个逻辑//tmp._day += day;//+不能改变自身值//while (tmp._day > GetMonthDay(tmp._year, tmp._month))//如果天数大于当前年月的天数//{//	tmp._day -= GetMonthDay(tmp._year, tmp._month);//当前天数减去当前月份的最大天数//	++tmp._month;//进入下一个月//	if (tmp._month == 13)//如果出现13月//	{//		++tmp._year;//进入下一年,月份归为一月//		tmp._month = 1;//	}//}return tmp;
}
Date Date::operator-(int day)
{Date tmp = *this;tmp -= day;return tmp;////-=复用-// Date tmp = *this;//if (day < 0)//获取需要减少的天数,入-50天//{//	return *this += -day;//}//tmp._day -= day;//当前天数减去需要减去的天数//while (tmp._day <= 0)//剩余天数小于0,退回上一个月,月数不足退回去年的12月,加上回退月数所有的天数,继续减去剩余天数,直到为正。//{//	--tmp._month;//	if (tmp._month == 0)//	{//		tmp._month = 12;//		tmp._year--;//	}//	// 借上⼀个⽉的天数//	tmp._day += GetMonthDay(tmp._year, tmp._month);// return tmp;一次拷贝//}
}
Date& Date::operator-=(int day)
{if (day < 0)//获取需要减少的天数,入-50天,则转换为正{return *this += -day;}_day -= day;//当前天数减去需要减去的天数while (_day <= 0)//剩余天数小于0,退回上一个月,月数不足退回去年的12月,加上回退月数所有的天数,继续减去剩余天数,直到为正。{--_month;if (_month == 0){_month = 12;_year--;}// 借上⼀个⽉的天数_day += GetMonthDay(_year, _month);}//-=复用-,相比-复用-=这里会多拷贝三次,增加开销.前两次是调用operator-=时两次拷贝和一次赋值拷贝//*this = *this - day;//相当于*this -= day;return *this;
}

这里实现了+、+=、-、-=四个运算符的重载,值得注意的是,在-和-=重写我注释掉了另外一种重写方法,两种方法是谁先重写就先复用谁,但是第二种方法会花费更大的开销。

//d1++ ;括号里数字随意,是整数就行
Date Date::operator++(int) {Date tmp = *this;*this += 1;return tmp;
}
//++d1
Date& Date::operator++() {*this += 1;return *this;
}

然后++的重载有些特殊,因为++分为++d和d++,在C++里规定,d++这种后置写法,重写需要带上参数,否则调用的是++d的重写方法。

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

这里我们传入两个日期,当max日期小于min日期时进行交换,揉捏在进行循环增加小的日期,直到两日期相等,我们则将重复增加多少次的次数返回,得到的就是两日期相差天数。flag是用于判断起始日期差是增加还是减少。

ostream& operator<<(ostream& out, const Date& d)
{out << d._year << "年" << d._month << "⽉" << d._day << "⽇" << endl;return out;
}
istream& operator>>(istream& in, Date& d)
{while (1) {cout << "请依次输⼊年⽉⽇:>";in >> d._year >> d._month >> d._day;if (!d.CheckDate()){cout << "⽇期⾮法" << endl;d.Print();cout << "重新输入" << endl;}}return in;
}

这里就是我们重写的输入输出流,方便我们直接输入年月日 并且以更直观的方式输出。

Date.cpp:

#include"Date.h"
bool Date::CheckDate()
{if (_month < 1 || _month > 12|| _day < 1 || _day > GetMonthDay(_year, _month))//不属于一到十二月,和1到月底日数{return false;}else{return true;}
}
Date::Date(int year, int month, int day)
{_year = year;_month = month;_day = day;if (!CheckDate()) {cout << "非法日期:" << endl;Print();}
}
void Date::Print()
{cout << _year << "-" << _month << "-" << _day << endl;
}
bool Date:: 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) {if (_day < d._day) {return true;}}else {return false ;}}/*return !(*this>=d);*///这里不用下面方式是因为至少有一个上述比较,否则会出现无限调用·死循环
}
//以此类推
bool Date:: operator <= (const Date& d) {return *this < d || *this == d;}bool Date:: operator >= (const Date& d) {return *this > d || *this == d;}
bool Date:: operator > (const Date& d) {//return *this < d || *this == 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);
}
//定义
Date& Date::operator+=(int day)
{	if (day < 0) {return *this -= (-day);}//当前天数加上输入天数_day += day;//这里是+=而不是=,原因在于改变了自身的值,+不能改变自身值while (_day > GetMonthDay(_year, _month))//如果天数大于当前年月的天数{_day -= GetMonthDay(_year, _month);//当前天数减去当前月份的最大天数++_month;//进入下一个月if (_month == 13)//如果出现13月{++_year;//进入下一年,月份归为一月_month = 1;}}return *this;
}
Date Date::operator+(int day)
{	//+可以使用拷贝构造Date tmp = *this;tmp += day;//函数复用,不用费那么多脑细胞再写一个逻辑//tmp._day += day;//+不能改变自身值//while (tmp._day > GetMonthDay(tmp._year, tmp._month))//如果天数大于当前年月的天数//{//	tmp._day -= GetMonthDay(tmp._year, tmp._month);//当前天数减去当前月份的最大天数//	++tmp._month;//进入下一个月//	if (tmp._month == 13)//如果出现13月//	{//		++tmp._year;//进入下一年,月份归为一月//		tmp._month = 1;//	}//}return tmp;
}
Date Date::operator-(int day)
{Date tmp = *this; //一次拷贝tmp -= day;return tmp;//一次拷贝////-=复用-// Date tmp = *this;//if (day < 0)//获取需要减少的天数,入-50天//{//	return *this += -day;//}//tmp._day -= day;//当前天数减去需要减去的天数//while (tmp._day <= 0)//剩余天数小于0,退回上一个月,月数不足退回去年的12月,加上回退月数所有的天数,继续减去剩余天数,直到为正。//{//	--tmp._month;//	if (tmp._month == 0)//	{//		tmp._month = 12;//		tmp._year--;//	}//	// 借上⼀个⽉的天数//	tmp._day += GetMonthDay(tmp._year, tmp._month);// return tmp;一次拷贝//}
}
Date& Date::operator-=(int day)
{if (day < 0)//获取需要减少的天数,入-50天,则转换为正{return *this += -day;}_day -= day;//当前天数减去需要减去的天数while (_day <= 0)//剩余天数小于0,退回上一个月,月数不足退回去年的12月,加上回退月数所有的天数,继续减去剩余天数,直到为正。{--_month;if (_month == 0){_month = 12;_year--;}// 借上⼀个⽉的天数_day += GetMonthDay(_year, _month);}//-=复用-,相比-复用-=这里会多拷贝三次,增加开销.前两次是调用operator-=时两次拷贝和一次赋值拷贝//*this = *this - day;//相当于*this -= day;return *this;
}//d1++ ;括号里数字随意,是整数就行Date Date::operator++(int) {Date tmp = *this;*this += 1;return tmp;}//++d1Date& Date::operator++() {*this += 1;return *this;}//前置后置推荐使用前置,后置多两次拷贝//日期相减int Date::operator-(const Date& d){Date max = *this;Date min = d;int flag = 1;if (*this < d){max = d;min = *this;flag = -1;}int n = 0;while (min != max){++min;++n;}return n * flag;}ostream& operator<<(ostream& out, const Date& d){out << d._year << "年" << d._month << "⽉" << d._day << "⽇" << endl;return out;}istream& operator>>(istream& in, Date& d){while (1) {cout << "请依次输⼊年⽉⽇:>";in >> d._year >> d._month >> d._day;if (!d.CheckDate()){cout << "⽇期⾮法" << endl;d.Print();cout << "重新输入" << endl;}}return in;}

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

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

相关文章

山西在职研究生培训机构综合评估:聚焦服务品质与特色课程竞争力深度剖析

随着在职人员学历提升需求持续增长,以及同等学力申硕报考人数的稳步上升,学员在选择培训机构时面临诸多困惑——如何辨别机构教学服务质量、如何匹配自身需求的课程体系、如何规避学习过程中的潜在风险,成为当下学员…

《R for Data Science (2e)》免费中文翻译 (第9章) --- Layers(1) - 教程

《R for Data Science (2e)》免费中文翻译 (第9章) --- Layers(1) - 教程2025-10-24 14:20 tlnshuju 阅读(0) 评论(0) 收藏 举报pre { white-space: pre !important; word-wrap: normal !important; overflow-x:…

2025年上海金蝶软件代理商推荐榜,聚焦服务品质与方案适配力深度剖析

随着数字经济加速渗透,上海中小企业数字化转型需求持续释放,金蝶软件作为主流 ERP 解决方案提供商,其代理商市场却呈现资质参差、服务能力分化的态势。企业在选型时常常面临 “方案与需求错配”“实施落地效率低”“…

nodejs和浏览器中事件循环机制

主要区别 nodejs中的事件循环分为六个阶段

2025年10月成都自动门厂家排名:五强服务与成本效益对比

如果您正在筹备写字楼、医院或商场的新建与改造项目,自动门系统往往是决定通行效率、节能表现与安全体验的第一道关口。成都作为西部商业与医疗设施升级最快的城市之一,2024年全市公共建筑自动门采购量同比增长18.6%…

2025年【介绍对象】贝特曼咨询:深度解析中美一体化移民服务风控体系

引言 本文聚焦“风控与合规”维度,拆解贝特曼咨询在EB-5投资移民全流程中的风险识别、隔离与缓释机制,为计划通过投资移民获取美国永久居留权的家庭提供一份可量化的客观参考。 背景与概况 贝特曼咨询“深植美国法律…

Shell脚本(.sh)函数语法

Shell脚本(.sh)函数语法函数返回值 1.范围(0-255) 2.实际数据通过echo输出并用$()捕获。 3.状态码返回(return) 通过$?获取上一函数的状态码 例如: my_func(){ return 42 } my_func echo $? #输出42 4.数据返回(…

playwright自动化测试应用-Day2- 【5种元素定位选择器】

优先级:测试ID > 角色选择器 > 文本选择器 > CSS 选择器 > xpath > 其他 1、首选-测试ID (最稳定)测试ID 是专门为自动化测试添加的标识属性,用于稳定地定位元素,不受U样式或布局变化的影…

Unity-动画IK控制 - 实践

pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", …

神曲

神曲Aloha Heja He Lendo calendo

2025年越南旅游旅行社推荐榜:聚焦企业服务品质与特色线路竞争力深度剖析

随着越南海滨度假与文化体验旅游热度持续攀升,以及岘港、下龙湾、富国岛等目的地的旅游需求增长,游客在选择旅游社时面临诸多困惑——如何辨别企业服务能力、如何匹配自身需求的特色线路、如何规避旅游过程中的潜在风…

2025年10月EB5投资移民中介权威榜:五强对比与选择指南

随着2025财年H-1B中签率跌至11.7%,越来越多留美家庭把“身份确定性”押注在EB-5投资移民通道。美国国务院签证办公室最新排期公告显示,中国大陆申请人10月表A前进一周,但预留签证仍无排期,窗口期转瞬即逝。与此同时…

2025年深圳印尼物流专线公司最新推荐榜,聚焦服务品质与线路竞争力深度剖析

随着印尼成为东南亚最大电商市场(预计 2025 年占区域市场规模 52%),中国与印尼的跨境贸易往来日益频繁,深圳作为物流枢纽,印尼专线服务需求持续攀升。企业在选择物流服务商时面临诸多困惑 —— 如何辨别企业跨境运…

2025年10月订货系统平台推荐:快马领衔实力榜对比

“订货系统平台”这六个字,对大多数商贸批发老板来说,就是“现金流”和“库存”之间的那根弦。10月正值秋冬备货高峰,渠道订单集中爆发,手工抄单、微信群接龙、Excel对账的老办法,常常把办公室变成“救火现场”:…

深度解析噬菌体展示抗体库:从文库构建到特定抗体筛选的核心逻辑

在噬菌体展示技术的应用体系中,噬菌体展示抗体库是实现 “高效筛选特定单克隆抗体片段” 的核心载体。它通过将海量不同特异性的抗体片段基因与噬菌体外壳蛋白基因融合,构建出 “每颗噬菌体展示一种抗体片段、每种抗…

2025年10月人形机器人场景落地商排行榜:赛飞特工程技术集团深度评测

如果您正在寻找能把人形机器人真正用到工厂、仓库、展厅或课堂里的落地商,大概率会被“演示很炫、落地很难”的落差困扰:设备到了现场识别率骤降、运维响应慢、课程配套缺位,预算因此打水漂。2025年工信部《人形机器…

2025年10月除甲醛空气净化器推荐榜:五强数据横向评测

进入2025年第四季度,全国多地陆续迎来供暖季,门窗密闭时间拉长,室内甲醛释放量随之抬升。中国室内装饰协会发布的《2025中国室内环境研究简报》指出,新装修住宅在供暖首月甲醛平均浓度较平时升高约42%,而老旧家具…

浙江AI快速建站公司口碑榜:聚焦技术革新、服务效率及企业成长力的综合测评

在数字化转型加速的背景下,AI快速建站服务成为企业提升线上竞争力的重要工具。浙江省作为科技创新活跃区域,涌现出一批以人工智能技术为核心的建站平台企业。本文基于技术实力、用户反馈、服务效率及企业成长性等多维…

数据库 单机 / 主备 / 主从 / 多主架构选择

数据库 单机 / 主备 / 主从 / 多主架构选择 数据库架构的选择需围绕业务优先级(如高可用、读写性能、数据一致性)、成本预算和维护复杂度展开。以下是单机、主备、主从、多主四种核心架构的详细对比及选择建议: 一、…

[随笔13] 日常杂事 - 枝-致

这些天跟ZN的关系越来越差,他开始言语侮辱我外加冷暴力。 上上周本来打算跟公婆分来住,就去看房子,在看房子的路上又发生激烈的争吵,主要是因为想起他的妈妈说我的那些话,我很生气,然后就开始骂他的妈妈和他。然…