C++類和對象實戰之Date類的實現方法
零、前言
在學瞭C++類和對象基本知識以及六個默認成員函數後,我們可以上手實現一個Date類出來,檢驗學習的效果。
一、Date類相關接口
接口展示:
class Date { //輸出操作符重載 friend ostream& operator<<(ostream& _cout, const Date& d); //輸出操作符重載 friend istream& operator>>(istream& _cin, Date& d); public: // 獲取某年某月的天數 int GetMonthDay(int year, int month); // 全缺省的構造函數 Date(int year=1988, int month=1, int day=1); // 拷貝構造函數 Date(const Date& d); // 賦值運算符重載 Date& 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--(); // >運算符重載 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); // 日期-日期 返回兩個日期之間相隔的具體天數 int operator-(const Date& d); //日期展示 void print() { cout << _year << " " << _month << " " << _day << endl; } private: int _year; int _month; int _day; };
二、具體接口函數實現
註意:
因為對於定義在類裡面的函數會自動設成內聯函數,而隻有一些簡單的函數才建議設成內聯函數,所以實現函數時我們是聲明和定義分離(在類裡面聲明,類外定義)
在類外實現函數接口需要加上類域名稱
1、獲取月份天數
註意:
閏年二月與平年二月的天數不同
實現代碼:
//獲取月份天數 int Date::GetMonthDay(int year, int month) { //設置平年月天數數組 static int monthdays[] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };//設置成靜態避免重復創建 int day = monthdays[month]; //對閏年二月的處理 if (month == 2 && ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0)) { day = 29; } return day; }
2、Date打印
註:打印函數比較簡單,設成內聯函數很適合,可以直接在類裡定義
實現代碼:
void Date::Print() { cout << _year << "年" << _month << "月" << _day << "日" << endl; }
3、Date構造函數
註意:
對於構造函數建議寫成全缺省函數(便於無參數初始化),但是隻能定義和聲明其中一個寫缺省
考慮初始化的日期是否合理
實現代碼:
//構造函數 //類裡聲明 Date(int year = 0, int month = 1, int day = 1); //定義 Date::Date(int year, int month, int day) { // 檢查日期的合法性 if (year >= 0 && month > 0 && month < 13 && day > 0 && day <= GetMonthDay(year, month)) { _year = year; _month = month; _day = day; } else { // 嚴格來說拋異常更好 cout << "非法日期" << endl; cout << year << "年" << month << "月" << day << "日" << endl; exit(-1); } }
4、Date析構函數
註:對於像Date一樣的類來說,析構函數(沒有需要清理的空間資源),拷貝函數和賦值重載函數(能夠完成成員變量淺拷貝)都不用自己寫,編譯器默認生成的已經足夠使用
實現代碼:
//析構函數 Date::~Date() { _year = 1; _month = 0; _day = 0; }
5、Date拷貝構造函數
實現代碼:
//拷貝構造 Date::Date(const Date& d) { _year = d._year; _month = d._month; _day= d._day; }
6、Date賦值重載函數
註意:
對於賦值操作符來說,是需要能支持連續賦值的操作,這裡我們返回Date本身來進行接下來的繼續賦值
實現代碼:
//賦值運算符重載 Date& Date::operator=(const Date& d) { _year = d._year; _month = d._month; _day = d._day; return *this; }
效果圖:
7、Date+=天數
註意:
- +=表示會修改Date本身的數據
- 處理傳入負數天數
- 處理好天數進位,月份進位
實現代碼:
//日期+=天數 Date& Date::operator+=(int day) { if (day < 0)//處理特殊情況 { *this -= -day;//復用Date-=天數 } else { _day += day; while (_day > GetMonthDay(_year, _month))//處理數據合理性 { _day -= GetMonthDay(_year, _month); _month++; if (_month > 12) { _year++; _month = 1; } } } return *this;//返回引用,即對象本身 }
8、Date+天數
註意:
+天數表示不會修改Date本身的數據(使用const修飾,避免修改)
邏輯與Date+=天數基本一致,可以進行復用
實現代碼:
Date Date::operator+(int day) const { Date tmp = *this;//賦值重載 tmp += day;//復用+=重載 return tmp;//返回值(拷貝構造) }
9、Date-=天數
註意:
- +=表示會修改Date本身的數據
- 處理傳入負數天數
- 考慮日期的借位,月份的借位
實現代碼:
//日期-=天數 Date& Date::operator-=(int day) { if (day < 0) { *this += -day;//復用Date+=天數 } else { _day -= day; while (_day <= 0)//處理數據合理性 { _month--; if (_month <= 0) { _year--; _month = 12; } _day += GetMonthDay(_year, _month); } } return *this; }
10、Date-天數
註意:
- -天數不會修改Date本身的數據(使用const修飾,避免修改)
- 邏輯與Date-=天數基本一致,可以進行復用
實現代碼:
Date Date::operator-(int day) const { Date tmp = *this; tmp -= day; return tmp; }
11、++Date
註意:
前置++表示,Date先增後使用
實現代碼:
//++Date Date& Date::operator++() { *this += 1;//復用Date+=天數 return *this; }
12、Date++
註意:
語法規定,因為與前置命名相同的緣故,這裡的後置函數多一個參數來與前置函數形成重載
後置++表示先使用後自增
實現代碼:
//Date++ Date Date::operator++(int) { Date tmp = *this;//保存一份日期 *this += 1;//自增當前日期 return tmp;//返回自增前的日期 }
13、–Date
實現代碼:
//--Date Date& Date::operator--() { *this -= 1; return *this; }
14、Date–
實現代碼:
//Date-- Date Date::operator--(int) { Date tmp = *this; *this -= 1; return tmp; }
15、日期比較
註:可以多次復用
實現代碼:
//日期比較 bool Date::operator>(const Date& d) const { 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) 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 { 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); }
16、Date相減
實現代碼:
//日期減日期 int Date::operator-(const Date& d) const { //確定日期的大小 Date max = *this; Date min = d; if (*this < d)//復用日期比較 { max = d; min = *this; } int day = 0; while (max != min) { ++min; ++day; } return day; }
17、日期輸入\日期輸出
註意:
- 對於輸入操作符,我們習慣是cin>>date,而這樣的用法表示做操作數是cin,右操作數為日期對象,但是對於類成員函數來說,存在著隱含參數this指針(占據和第一個參數位置,即日期對象是左操作數)
- 雖然定義成類外函數能修改參數位置,但是無法訪問類裡的私有成員變量,這裡我們使用友元函數來解決,即在類裡聲明函數前加上friend,便可以訪問成員
實現代碼:
//輸出操作符重載 ostream& operator<<(ostream& _cout, const Date& d) { _cout << d._year << "年" << d._month << "月" << d._day << "日" ; return _cout; } //輸出操作符重載 istream& operator>>(istream& _cin, Date& d) > { _cin >> d._year >> d._month >> d._day; return _cin; }
效果圖:
- date,而這樣的用法表示做操作數是cin,右操作數為日期對象,但是對於類成員函數來說,存在著隱含參數this指針(占據和第一個參數位置,即日期對象是左操作數)
- 雖然定義成類外函數能修改參數位置,但是無法訪問類裡的私有成員變量,這裡我們使用友元函數來解決,即在類裡聲明函數前加上friend,便可以訪問成員
實現代碼:
//輸出操作符重載 ostream& operator<<(ostream& _cout, const Date& d) { _cout << d._year << "年" << d._month << "月" << d._day << "日" ; return _cout; } //輸出操作符重載 istream& operator>>(istream& _cin, Date& d) { _cin >> d._year >> d._month >> d._day; return _cin; }
效果圖:
總結
到此這篇關於C++類和對象實戰之Date類實現的文章就介紹到這瞭,更多相關C++ Date類的實現內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!