从C语言到C++⑥(第二章_类和对象_中篇_续)大练习(日期类)+笔试选择题(上):https://developer.aliyun.com/article/1513649
1.4 日期 -= 天数 和 日期 - 天数
我们刚才实现了加和加等 ,现在我们来实现一下减和减等。
加和加等是进位,那减和减等自然就是借位。
我们先把day减一下天数,此时如果日期不合法,那我们就需要进行借位操作。
和上面一样我们先实现减等:
Date& Date::operator-=(int day) { _day -= day; while (_day <= 0) { _month--; if (_month == 0) { _year--; _month = 12; } _day += GetMonthDay(_year, _month); } return *this; }
如果减完后的天数小于等于 0,就进入循环,向月 "借位" ,
因为已经借出去了,所以把 月份 -- 。还要考虑月份会不会借完的情况,
月份为 0 的时候就是没得借了,这种情况就向年借,年--,
之后加上通过 GetMonthDay 获取当月对应的天数,就是所谓的 "借",
循环继续判断,直到天数大于0的时候停止,返回 *this 。
出了作用域 *this 还在,所以我们可以使用引用返回 Date& 。
一样的,我们复用一下 -= 就可以把 - 实现出来了:
Date Date::operator-(int day) const { Date ret = *this; ret -= day;// ret.operator-=(day); return ret;// 和+一样,出了作用域ret对象就不在了,所以不能用引用返回 }
到这我们发现日期计算器是可以输入负数的,那我们的输入负数呢:
我们发现,在设计 operator-= 的时候是 <= 0 才算非法的,所以这种情况就没考虑到。
我们可以这么设计,在减天数之前对 day 进行一个特判,
因为你减负的100就相当于加正的100,就变成加了,
Date& Date::operator-=(int day) { if (day < 0) { return *this += -day; } _day -= day; while (_day <= 0) { _month--; if (_month == 0) { _year--; _month = 12; } _day += GetMonthDay(_year, _month); } return *this; }
我们再回去把 operator+= 处理一下:
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) { _year++; _month = 1; } } return *this; }
1.5 日期++ 和 ++日期
d1++; ++d1;
因为都是 operator++ ,为了能让编译器直到我们实现的到底是 前置++ 还是 后置++,
这里就到用到一个叫做 "参数占位" 的东西 :int ,即:
后置++ 带 " int " ,构成函数重载。
operator++(int); // 带int,表示后置加加 d1++ operator++(); // 不带, 表示前置加加 ++d1
根据上面的经验,我们先实现返回运算符作用后的值的前置++
因为 前置++ 返回的是加加之后的值,所以我们使用引用返回:
Date& Date::operator++() // 前置 { return *this += 1; }
这里我们直接复用 +=,加加以后的值就是 *this ,我们返回一下 *this 就行。
因为后置++返回的是加加之前的值,所以我们不用引用返回:
Date Date::operator++(int) // 后置 { Date ret = *this; *this += 1; return ret; }
这里要拷贝构造两次,所以我们推荐以后自定义类型++,使用前置++ 。
对于内置类型,用前置比后置也好那么一丢丢,但现在几乎可忽略不计,
为了和自定义类型对应好看点,我们以后对于内置类型也尽量用前置好了
1.6 日期-- 和 --日期
和 operator++ 一样,operator-- 为了能够区分前置和后置,也要用 int 参数占位
后置-- 带 " int " ,构成函数重载。
前置-- 返回的是减减之后的值,所以我们使用引用返回:
Date& Date::operator--() // 前置 { return *this -= 1; }
后置-- 返回的是减减之前的值,所以我们不用引用返回,
我们在减减之前先拷贝构造一个 "替身" ,待本体加加后,
把替身 ret 返回回去,就实现了返回减减之前的值:
Date Date::operator--(int) // 后置 { Date ret = *this; *this -= 1; return ret; }
和后置++一样这里要拷贝构造两次,所以我们推荐以后自定义类型--,使用前置-- 。
1.7 日期 - 日期 operator-
我们刚才实现的 operator- 是日期减天数的:
Date Date::operator-(int day) const { Date ret = *this; ret -= day;// ret.operator-=(day); return ret;// 和+一样,出了作用域ret对象就不在了,所以不能用引用返回 }
日期 + 日期 没什么意义,但是日期 - 日期还是有意义的,
如果我们想要计算一下今天距离高考还有多少天呢?
那我们就需要写一个日期减日期版本的 operater -
因为有正用负,我们利用学二叉树时讲过的思想吧
之后我们用计数器的方式来实现就可以了 :
int Date::operator-(const Date& d) const { int ret = 0; int flag = -1; Date min = *this;//默认第一个小,返回的时候乘上 -1 Date max = d; if (*this > d)//默认错误,把小和大重置,返回时乘上 1 { flag = 1; min = d; max = *this; } while (min != max) { ++min; ++ret; } return ret * flag; }
1.8 打印*this是星期几
前面我们使用日期计算器得出一个日期的时候,它都会给我们那个日期是星期几,
我们也来实现一个类似的功能。
我们需要找一个标志日期,知道那一天是星期几,然后复用上面的代码就能实现这个功能,
因为十六世纪有日期被抹掉了,所以我们拿 1900年1月1号(刚好是星期一) 作为起始点,
和上面一样创建一个const数组,
两个日期相减得到的结果,%7 作为下标去访问 Week 数组里我们已经准备好的周几,
打印出来就可以了:
void Date::PrintWeekDay() const //打印*this是星期几 { const char* Week[] = { "星期一","星期二" ,"星期三" , "星期四" ,"星期五" , "星期六" , "星期天" }; Date flag(1900, 1, 1); //1900年1月1日是星期一,自己减自己为0,对应下标0 cout << Week[(*this - flag) % 7] << endl; }
从C语言到C++⑥(第二章_类和对象_中篇_续)大练习(日期类)+笔试选择题(下):https://developer.aliyun.com/article/1513651?spm=a2c6h.13148508.setting.30.5e0d4f0eimuY68