运算符重载和const成员
- const成员
- const修饰类成员函数
- const对象调用权限
- 小结
- 运算符重载
const成员
const成员函数:const修饰的成员函数。const修饰类成员函数,实际限制的是*this,表明该成员函数不能对类的任何成员进行修改。
const修饰类成员函数
const对象调用权限
小结
- const对象不能调用非const成员函数。
- 非const对象可以调用const成员函数。
- const成员函数内可以调用其他的非const成员函数。
- 非const成员函数内可以调用其他的const成员函数。
结论:
- 成员函数后面+const以后,普通和const对象都可以调用。
- 不是所有成员函数都可以+const,要修改对象成员变量的函数不能+
- 只要成员函数内部不修改成员变量,都应该+const,这样const对象和普通对象都可以调用
运算符重载
借用日期类,实现运算符的重载。为了便于查看,把成员函数采用声明和定义分离的形式,写在两个文件中。
Date.h:
#include <iostream>
#include <assert.h>
using namespace std;
class Date
{
//友元函数声明
friend ostream& operator<<(ostream& out, Date& d);
friend istream& operator>>(istream& in, Date& d);
public:
//全缺省构造函数
Date(int year = 1970, int month = 1, int day = 1);
//打印日期
void Print() const
{
cout << _year << "-" << _month << "-" << _day << endl;
}
//拷贝构造函数
//d2(d1)
Date(const Date& d)
{
_year = d._year;
_month = d._month;
_day = d._day;
}
//运算符重载
//d2 = d3; <==>d2.operator(d3)
Date& operator=(const Date& d);
//析构函数
~Date();
// < 运算符重载
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;
//获取某年某月的天数
int GetMonthDay(int year, int month);
//日期+=天数
Date& operator+=(int day);
//日期+天数
Date operator+(int day) const;
//前置++
Date& operator++();
//后置++
Date operator++(int);
//日期-=天数
Date& operator-=(int day);
//日期-天数
Date operator-(int day) const;
//前置--
Date& operator--();
//后置--
Date operator--(int);
//日期-日期 返回天数
int operator-(const Date& d) const;
private:
int _year;
int _month;
int _day;
};
ostream& operator<<(ostream& out, Date& d);
istream& operator>>(istream& in, Date& d);
Date.cpp:
#include "Date.h"
//构造函数
Date::Date(int year, int month, int day)
{
if (month > 0 && month < 13
&& day > 0 && day <= GetMonthDay(year, month))
{
_year = year;
_month = month;
_day = day;
}
else
{
cout << "非法日期" << endl;
assert(false);
}
}
//运算符重载
Date& Date::operator=(const Date& d)
{
if (this != &d)
{
_year = d._year;
_month = d._month;
_day = d._day;
}
return *this;
}
//析构函数
Date::~Date()
{
_year = 0;
_month = 0;
_day = 0;
}
// < 运算符重载
bool Date::operator<(const Date& x) const
{
if (_year < x._year)
{
return true;
}
else if (_year == x._year && _month < x._month)
{
return true;
}
else if (_year == x._year && _month == x._month && _day < x._day)
{
return true;
}
return false;
}
// == 运算符重载
bool Date::operator==(const Date& x) const
{
return _year == x._year
&& _month == x._month
&& _day == x._day;
}
// 复用
// <= 运算符重载
bool Date::operator<=(const Date& x) const
{
return *this < x || *this == x;
}
// > 运算符重载
bool Date::operator>(const Date& x) const
{
return !(*this <= x);
}
// >= 运算符重载
bool Date::operator>=(const Date & x) const
{
return !(*this < x);
}
// != 运算符重载
bool Date::operator!=(const Date& x) const
{
return !(*this == x);
}
//获取某年某月的天数
int Date::GetMonthDay(int year, int month)
{
static int daysArr[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
//if (((year % 4 == 0 && year % 100 != 0) || (year % 400 == 0)) && month == 2)
if (month == 2 && ((year % 4 == 0 && year % 100 != 0) || (year % 400 == 0)))
{
return 29;
}
else
{
return daysArr[month];
}
}
//日期+=天数
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;
}
// d1 + 100
//日期+天数
Date Date::operator+(int day) const
{
Date tmp(*this);
tmp += day;
return tmp;
}
//日期-=天数
Date& Date::operator-=(int day)
{
if (day < 0)
{
return *this += -day;
}
_day -= day;
while (_day <= 0)
{
--_month;
if (_month == 0)
{
_month = 12;
--_year;
}
_day += GetMonthDay(_year, _month);
}
return *this;
}
//日期-天数
Date Date::operator-(int day) const
{
Date tmp = *this;
tmp -= day;
return tmp;
}
// 前置++
Date& Date::operator++()
{
*this += 1;
return *this;
}
// 后置++
// 增加这个int参数不是为了接收具体的值,仅仅是占位,跟前置++构成重载
Date Date::operator++(int)
{
Date tmp = *this;
*this += 1;
return tmp;
}
//前置--
Date& Date::operator--()
{
*this -= 1;
return *this;
}
//后置--
Date Date::operator--(int)
{
Date tmp = *this;
*this -= 1;
return tmp;
}
//日期-日期 返回天数
// d1 - d2;
int Date::operator-(const Date& d) const
{
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)
{
int year, month, day;
in >> year >> month >> day;
if (month > 0 && month < 13
&& day > 0 && day <= d.GetMonthDay(year, month))
{
d._year = year;
d._month = month;
d._day = day;
}
else
{
cout << "非法日期" << endl;
assert(false);
}
return in;
}