实现上一些细节:
这里由于自转和公转问题,当是闰年时,二月的天数加一 还有一些细节上的问题(但是CPU跑太快,没啥影响),static int montharr
属于静态变量,只能定义一次。对此频繁调用时,不用多次定义。 在判断语句中,可以将位置进行调正,这里跟&&短路知识点有关,如果前面是假,不同接下去判断,整个表达式都为假
三、比较两个日期
这里需要涉及到运算符重载,这里有个小技巧,只需要实现大于等于或小于等于的接口。
函数声明 :
bool operator == ( const Date& d) const ;
bool operator != ( const Date& d) const ;
bool operator < ( const Date& d) const ;
bool operator <= ( const Date& d) const ;
bool operator > ( const Date& d) const ;
bool operator >= ( const Date& d) const ;
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
实现小于等于(剩下可取反) :
bool Date:: operator == ( const Date& d) const
{
return _year == d. _year && _month == d. _month && _day == d. _day;
}
bool Date:: operator != ( const Date& d) const
{
return ! ( * this == d) ;
}
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 && _day < d. _day)
return true ;
else
return false ;
}
bool Date:: operator <= ( const Date& d) const
{
return ( * this < d) || ( * this == d) ;
}
bool Date:: operator > ( const Date& d) const
{
return ! ( * this <= d) ;
}
bool Date:: operator >= ( const Date& d) const
{
return ! ( * this < d) ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
四、日期加天数
这里需要考虑持续进位原则
Date& Date:: operator += ( int day)
{
_day += day;
while ( _day > GetMonthDay ( _year, _month) )
{
_day -= GetMonthDay ( _year, _month) ;
++ _month;
if ( _month == 13 )
{
++ _year;
_month = 1 ;
}
}
return * this ;
}
Date Date:: operator + ( int day) const
{
Date temp ( * this ) ;
temp += day;
return temp;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
当日期加完天数后,通过日期的规则需要按照进位原则,对年月日数据进行调正 在实现operator+=/+,都可以间接实现operator+/+= 这里operator+=使用日期加天数,提高了效率和避免传值返回中的拷贝过程 operator+这里不能使用引用返回,这里是创建了一个临时变量,调用完会销毁
4.1 先实现operator+还是operator+=
先实现operator+=,再间接实现operator+呢? 还是先实现operator+,再间接实现operator+=呢?
对此得出结论 :推荐先实现operator+=,再间接实现operator+。
理由:
将代码贴出来,方便进行对比。这里不能交叉对比,需要采用横向对比。 两者实现operator+是等价,但是实现operator+=,左边需要复用+operator,相比之下多次拷贝构造。 这不详细介绍关于operator-和operator-=。大体逻辑跟operator+和operator+有异曲同工之处(具体在Date.cpp看看)
五、日期减日期
需要使用持续借位原则,如果天数为0,需要得到上月的天数
第一种方式 :不断++直到等于大的那个年份,这里需要日期加日期接口的技术支撑
int Date:: operator - ( const Date& d)
{
int flag = 1 ;
Date max = * this ;
Date min = d;
if ( * this < d)
{
int flag = - 1 ;
max = d;
min = * this ;
}
int n = 0 ;
while ( min != max)
{
++ min;
++ n;
}
return n * flag;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
第二个方法 :通过该变量记录两个年份修饰到1月1日,也就是都修饰到1月1日之间还差多少天,再计算两个年之间有多少个年,如果是平年+365,闰年+366
int Date:: operator - ( const Date& d) const
{
Date max = * this ;
Date min = d;
int flag = 1 ;
if ( * this < d)
{
Date max = d;
Date min = * this ;
int flag = - 1 ;
}
int count = 0 ;
while ( ! ( max. _day == 1 && max. _month == 1 ) )
{
-- max;
++ count;
}
while ( ! ( min. _day == 1 && min. _month == 1 ) )
{
-- min;
-- count;
}
while ( min. _year != max. _year)
{
if ( is_leapyear ( min. _year) )
count += 366 ;
else
count += 365 ;
++ min. _year;
}
return flag * count;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
六、流插入和流提取运算符重载
out和cin的本质是输入和输出流对象 ,对于<<和>>用于重载的运算符,从图可以得,cout
属于ostream
类,cin
属于istream
类,可以自动识别类型
对于我们可以在日期类中,实现<<和>>重载打印日期和提取日
int main ( )
{
Date d1 ( 2024 , 3 , 10 ) ;
如果使用运算符重载,隐含的this指针占用第一个参数的位置,Date必须是左操作数,d1<是不符合我们的习惯的
对此我们可以在类外实现该运算符重载函数,就可以自己设计参数部分的位置 但是又引出了另一个问题:类外不能访问类中的私有成员 ,如果将私有权限放开,就缺乏安全性,对此C++中有友元,接下来我们会涉及到,这里就使用下,虽然这个全局函数不在类中,但是可以访问私有成员函数
friend ostream& operator << ( ostream& out, const Date& d) ;
friend istream& operator >> ( istream& in, Date& d) ;
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;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
如果我们需要连续打印cout<的话,这里就不合适的。因为这里的结合性是从左往右,cout<会返回一个临时变量,那么这里运算符重载函数需要通过引用返回了
。C++存在私有的,printf
不支持自定义打印,cout
本质实现所用类型的打印。
- * * 用引用做返回值,应对连续流插入和流提取
- 流提取不是不能对Date进行const 修饰,需要通过键盘读取数据存储在成员变量
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
6.1 需要判断输入进去的数据是否有误
七、源代码展示
7.1 Date.h
# pragma once
# include
# include
using namespace std;
class Date
{
public :
Date ( int year = 1 , int month = 1 , int day = 1 ) ;
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 ) ;
Date operator -- ( int ) ;
Date& operator -- ( ) ;
int operator - ( const Date& d) ;
int GetMonthDay ( int year, int month)
{
assert ( month > 0 && month < 13 ) ;
static int monthDays[ 13 ] = { 0 , 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] ;
}
void Print ( )
{
cout << _year << "/" << _month << "/" << _day << endl;
}
private :
int _year;
int _month;
int _day;
} ;
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61
7.2 Date.cpp
# include "Date.h"
Date :: Date ( int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
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 ;
}
}
}
return false ;
}
bool Date:: operator <= ( const Date& d)
{
return * this < d || * this == d;
}
bool Date:: operator > ( const Date& d)
{
return ! ( * this <= d) ;
}
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) ;
}
Date& Date:: operator += ( int day)
{
_day += day;
while ( _day > GetMonthDay ( _year, _month) )
{
_day -= GetMonthDay ( _year, _month) ;
++ _month;
if ( _month == 13 )
{
++ _year;
_month = 1 ;
}
}
return * this ;
}
Date Date:: operator + ( int day)
{
Date tmp = * this ;
tmp += day;
return tmp;
}
Date Date:: operator - ( int day)
{
Date tmp = * this ;
tmp -= day;
return tmp;
}
Date& Date:: operator -= ( int day)
{
_day -= day;
while ( _day <= 0 )
{
-- _month;
if ( _month == 0 )
{
-- _year;
_month = 12 ;
}
_day += GetMonthDay ( _year, _month) ;
}
return * this ;
}
Date& Date:: operator ++ ( )
{
* this += 1 ;
return * this ;
}
Date Date:: operator ++ ( int )
{
Date tmp = * this ;
* this += 1 ;
return tmp;
}
int Date:: operator - ( const Date& d)
{
int flag = 1 ;
Date max = * this ;
Date min = d;
if ( * this < d)
{
int flag = - 1 ;
max = d;
min = * this ;
}
int n = 0 ;
while ( min != max)
{
++ min;
++ n;
}
return n * flag;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
7.3 test.cpp
# define _CRT_SECURE_NO_WARNINGS 1
# include "Date.h"
int main ( )
{
Date d1 ( 2024 , 1 , 29 ) ;
Date d2 = d1 + 20 ;
d2. Print ( ) ;
d1. Print ( ) ;
d2 -= 20 ;
d2. Print ( ) ;
d1 += 30000 ;
d1. Print ( ) ;
++ d1;
d1. operator ++ ( ) ;
d1. Print ( ) ;
d1++ ;
d1. operator ++ ( 10 ) ;
d1. Print ( ) ;
Date d4 ( 2024 , 1 , 29 ) ;
Date d5 ( 2024 , 8 , 1 ) ;
cout << d5 - d4 << endl;
return 0 ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
以上就是本篇文章的所有内容,在此感谢大家的观看!这里是店小二呀C++笔记,希望对你在学习C++语言旅途中有所帮助!
data-report-view="{"mod":"1585297308_001","spm":"1001.2101.3001.6548","dest":"https://blog.csdn.net/2302_79177254/article/details/141017550","extend1":"pc","ab":"new"}">>
评论记录:
回复评论: