3_V1-類和對象 -- 默認成員函數

定義一個日期類

#include <iostream>
#include <assert.h>
using namespace std;class Date
{
public:void Display();
private:int _year;int _month;int _day;
};

注意:

在定義一個類的時候往往會將其成員變量定義為私有,成員函數定義為公有.這是為了達到軟件工程上的高內聚低耦合的要求

this指針

每一個函數都有自己的形式指針,它的名字是固定的,稱為this,同時this指針是隱式的.
編譯器會對成員函數進行優化,在對象調用成員函數的時候,此時會將該對象的地址傳遞給成員函數的第一個參數
同時this指針是成員函數隱含指針參數,我們程序員不能自己將其加到成員函數的形式參數列表中,也不能在調用的時候顯示的將對象的地址傳給this指針
這里寫圖片描述

注意:

this指針是形式參數,既然是形式參數那么它就會在棧上,但是在Linux下,this指針是被存放在寄存器中.同時誰來調用this成員函數那么this指針就指向誰

構造函數

構造函數可以達到原子性,即對象被定義的時候就會被初始化.在定義類的時候我們呢看到通常會將成員函數定義為私有但是私有成員變量在類外是不能被訪問的.為了對成員函數進行初始化,此時就需要構造函數來對私有成員變量進行初始化.同時構造函數只在對象被定義的時候僅僅被執行一次.下面來說一下構造函數的特點
1.函數名和類名相同
2.無返回值
3.對象構造時系統自己調用對應的構造函數
4.構造函數可以重載(函數名相同, 參數不同)
5.構造函數可以在類里面定義也可以在類外面定義
6.如果類中沒有定義一個構造函數,系統會默認生成一個缺省的構造函數,但是當我們定義了構造函數,此時系統就不會生成默認的缺省構造函數,而會定義我們自己定義的構造函數,編譯器在對其進行初始化的時候,內置的類型會被初始化,但是自定義的不會進行初始化
7.無參的構造函數和全缺的構造函數都認為是缺省構造函數.并且缺省的構造函數只能有一個

幾種構造函數

class Date
{
public:// 1.無參的構造函數// Date()// {// }// 2.缺省構造函數Date(int year = 1900, int month = 1, int day = 1){if(year < 1900 || month < 0 || month > 12 || day < 0 || day > 31){assert(0);}_year = year;_month = month;_day = day;}//3. 帶參的構造函數// Date (int year, int month, int day)// {//     _year = year;//     _month = month;//     _day = day;// }// 4. 拷貝構造函數,創建一個對象Date(Date& d){_year = d._year;_month = d._month;_day = d._day;}void Display();
private:int _year;int _month;int _day;
};

注意:

拷貝構造其實就可以理解為一種特殊的構造函數,它是構造函數的重載,同時注意拷貝構造是對象的創建,而我們后面將所說的賦值是兩個對象都已經存在
同時拷貝構造必須傳引用,因為如果傳值的話那么就會出現無窮遞歸(拷貝構造就要傳參,傳參就要拷貝構造)
3.如果自己沒有顯示進行定義,那么系統就會自己生成默認缺省構造函數.缺省的時候拷貝構造函數就會依次拷貝類成員進行初始化

析構函數

1.和類的名字前面加上一個~
2.無返回值
3.一個類有且只有一個析構函數(因為沒有參數,不能構成重載),如果我們自己不寫編譯器會自動生成
4.對象生命周期結束的時候,系統調用析構函數
5.析構函數不能刪除對象,只是對對象進行清理

注意:

如果需要清理成員變量(動態開辟空間),則自己寫析構函數,對象在定義的時候一定調用了構造,生命周期結束的時候就一定調用了析構函數

運算符重載

Date& operator = (const Date& d){this -> _year = d._year;this -> _month = d._month;this -> _day = d._day;return *this;}// d2 == d3// d2.operator(*this, d3)bool operator == (const Date& d) {if(this -> _year == d._year && this -> _month == d._month && this -> _day == _day){return true;}return false;}bool operator != (const Date d){if(!(*this == d)){return true;}return false;}//d1 > d2//d1.operator > (this, d2)bool operator > (const Date& d) {if(this -> _year > d._year){return true;}if(this -> _year == d._year){if(this -> _month > d._month){return true;}}if(this -> _month == d._month){if(this -> _day > d._day){return true;}}return false;}//d2 < d3//d2.operator(this, d3)bool operator < (const Date& d) {if(*this == d || *this > d){return false;}return true;}// d2 >= d3bool operator >= (const Date& d){if(*this > d || *this == d){return true;}return false;}//d2 <= d3bool operator <= (const Date& d){if(*this < d || *this == d){return true;}return false;}//d2++Date& operator ++ () // 前置 {++( this -> _day );if(this -> _day > GetMonthDay(this -> _year, this -> _month)){this -> _day = 1;++(this -> _month);if((this -> _month) > 12){++(this -> _year);this -> _month = 1;}}return *this;}//d2++Date operator ++ (int) // 后置 {Date ret = (*this);++( *this );return ret;}//2018-1-1Date& operator -- (){--(this -> _day);if(this -> _day == 0){(this -> _month)--;if(this -> _month == 0){this -> _year -= 1;this -> _month = 12;}(this -> _day) += GetMonthDay(this -> _year, this -> _month);}return *this;}Date operator -- (int)//后置{Date ret = *this;--(*this);return ret;}Date& operator += (int day){this -> _day = this -> _day + day;while(this -> _day > GetMonthDay(this -> _year, this -> _month)){this -> _day = this -> _day - GetMonthDay(this -> _year, this -> _month);this -> _month ++;if(this -> _month > 12){this -> _month = 1;this -> _year ++;}}return *this;}//d1 + 30//d1.operator(this, day)Date operator + (int day) {Date ret = *this;ret += day;return ret;}Date& operator -= (int day){this -> _day -= day;while(this -> _day <= 0){this -> _month --;if(this -> _month < 1){this -> _month = 12;this -> _year --;}this -> _day += GetMonthDay(this -> _year, this -> _month);}return *this;}Date operator - (int day){Date ret = *this;ret -= day;return ret;}// 2018-1-31  -  2018-1-1// d1.operator(this, d2)int operator - (const Date& d){int count = 0;//記錄天數while(*this > d){--(*this);( *this ).Display();++count;}return count;}void Display()//展示日期類{cout<<_year<<"-"<<_month<<"-"<<_day<<endl;}int GetMonthDay(int year, int month){int day[13] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};if(( year %4 == 0 && year % 100 != 0 ) || ( year % 400 == 0 )){day[month] = day[2] + 1;}return day[month];}

輸入探索構造函數

類的成員函數有兩種初始化方式
1.初始化列表
以一個冒號開始, 接著一個逗號分隔數據列表, 每個數據成員都在括號里進行初始化. 盡量使用初始化列表, 因為初始化列表更加高效.
2.構造函數體內進行賦值
這里寫圖片描述
初始化列表為什么更加高效
1.初始化列表不寫, 編譯器會自動走一次(自定義成員變量)
2.初始化列表可以認為是成員變量定義的地方
3.構造函數體內賦值會產生臨時變量, 但是初始化列表不會產生臨時變量

那些成員變量必須放在初始化列表中

1.常量成員變量(常量創建時必須初始化)
2.引用類型成員變量(引用成員變量創建時必須初始化)
3.沒有缺省構造函數的自定義類型
4.成員變量初始化的時候按照聲明順序依次初始化,而非初始化列表出現的順序

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/383919.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/383919.shtml
英文地址,請注明出處:http://en.pswp.cn/news/383919.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

C++ 類模板二(類模版與友元函數)

http://www.cnblogs.com/zhanggaofeng/p/5661829.html //類模版與友元函數 #include<iostream> using namespace std;template<typename T> class Complex{ public:Complex(T a,T b);void Print() const//const修飾的是this指針{cout << this->Real <&…

HDU - 2973威爾遜定理

核心問題就是那個等式 我們觀察到等式可以寫成(n-1)!-1/n-[(n-1)!/n]的形式&#xff0c;這樣就應該聯想到威爾遜定理了。 回顧一下威爾遜定理的內容&#xff1a;當且僅當n為素數的時候n|(n-1)!-1&#xff0c;n為合數且大于4的時候n|(n-1)!【參見威爾遜定理的證明】 對于這個…

linux網絡編程之posix 線程(四):posix 條件變量與互斥鎖 示例生產者--消費者問題

http://blog.csdn.net/jnu_simba/article/details/9129939 一、posix 條件變量 一種線程間同步的情形&#xff1a;線程A需要等某個條件成立才能繼續往下執行&#xff0c;現在這個條件不成立&#xff0c;線程A就阻塞等待&#xff0c;而線程B在執行過程中使這個條件成立了&#x…

3-V2-類和對象 -- const內聯 靜態成員 友元

const修飾成員函數 在成員函數后面加一個const, const修飾this指針指向的對象, 保證調用這個const成員函數的對象在函數內不會被改變 注意:成員函數如果不修改成員變量,則加上const,成員函數如果要修改成員變量,此時就不能給其加上const修飾了 1.const對象不能調用非const…

C語言 二級指針內存模型混合實戰

http://www.cnblogs.com/zhanggaofeng/p/5485833.html //二級指針內存模型混合實戰 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> #include <string.h>//將內存模型①和內存模型②的數據拷貝到內存模型③ char ** threemodel(ch…

擴展歐幾里得算法

對于a?xb?yca*xb*yca?xb?yc,這樣一個二元一次方程組&#xff0c;我們想要得到他的一組解可以用擴展歐幾里得算法&#xff0c;參數列表的a,b,x,y就是方程中的a,b,x,y&#xff0c;d計算出來是gcd(a,b)。 算法求出來的是a?xb?ygcd(a,ba*xb*ygcd(a,ba?xb?ygcd(a,b的一組解…

Linux 網絡編程八(epoll應用--大并發處理)

http://www.cnblogs.com/zhanggaofeng/p/5901316.html //頭文件 pub.h #ifndef _vsucess#define _vsucess#ifdef __cplusplus extern "C" {#endif //服務器創建socket int server_socket(int port);//設置非阻塞 int setnonblock(int st);//接收客戶端socket int ser…

約瑟夫問題

n個人編號為0…n-1圍成一個圈,從0開始報數,每經過k個人那個人就退出這個圈不再報數,問最后留下來的人的編號. 樸素的做法當然是模擬,但是n,k的值一旦變得比較大的時候就難以解決問題. 我們考慮歸納的解決問題 當只有一個人的時候答案顯然為0, 假設我們已知n-1個人的時候答案為…

【數據結構與算法】內部排序之三:堆排序(含完整源碼)

轉載請注明出處&#xff1a;http://blog.csdn.net/ns_code/article/details/20227303 前言 堆排序、快速排序、歸并排序&#xff08;下篇會寫這兩種排序算法&#xff09;的平均時間復雜度都為O&#xff08;n*logn&#xff09;。要弄清楚堆排序&#xff0c;就要先了解下二叉堆這…

模線性方程(中國剩余定理+擴展中國剩余定理)

已知一系列除數和模數,求最小的滿足條件的數 我們先考慮一般的情況&#xff0c;即模數不互質。&#xff08;擴展中國剩余定理&#xff09; 我們考慮兩個方程的情況 x%MR xk1?MRxk1 * MRxk1?MR x%mr xk2?mrxk2 * mrxk2?mr 所以k1?MRk2?mrk1 * MRk2 * mrk1?MRk2?mr 即…

C++:Vector和List的實現

Vector的實現 //test.h #pragma once#include <iostream> #include <cstdio> #include <string.h> #include <assert.h>using namespace std;typedef int DataType;#define TESTHEADER printf("\n%s\n", __FUNCTION__)class Vector { publi…

【數據結構】(面試題)使用兩個棧實現一個隊列(詳細介紹)

http://blog.csdn.net/hanjing_1995/article/details/51539578 使用兩個棧實現一個隊列 思路一&#xff1a; 我們設定s1是入棧的&#xff0c;s2是出棧的。 入隊列&#xff0c;直接壓到s1即可 出隊列&#xff0c;先把s1中的元素倒入到s2中&#xff0c;彈出s2中的棧頂元素&#x…

POJ 1006 Biorhythms

中國剩余定理的模板題 只是有一個問題就是求出來Xk*MR中的R比給定的日期還大&#xff0c;但是如果負數的整除就不是向下取整了&#xff0c;為了解決這個問題&#xff0c;我們將R減小M&#xff0c;這樣總是正的&#xff0c;求出來的就沒有什么問題。 #include <iostream>…

POJ 3696 歐拉函數+快速冪

題目的意思大概就是問是否存在一串全是8的數字是L的倍數 直接想沒有什么想法&#xff0c;要想到用簡潔的形式將這個數字表示出來&#xff0c;對于每一位都是8的數字我們可以用 X8*(10k-1)/9的形式表示出來&#xff0c;那么題目的意思就是求X使L|X&#xff0c;我們先處理一下8和…

兩個棧實現一個隊列,兩個隊列實現一個棧

http://blog.csdn.net/zw_1510/article/details/51927554 問題1&#xff1a;用兩個棧實現一個隊列&#xff0c;實現隊列的push和delete操作 棧的特性是先進后出&#xff08;FILO&#xff09;,隊列的特性是先進先出&#xff08;FIFO&#xff09;,在實現delete時&#xff0c;我們…

C++:String的寫時拷貝

String的寫時拷貝 //test.h #pragma once#include <iostream> #include <string.h> #include <cstdio> #include <assert.h> using namespace std;#define TESTHEADER printf("\n%s\n", __FUNCTION__) class String { public:String(const …

兩個棧實現一個隊列與兩個隊列實現一個棧

http://blog.csdn.net/z84616995z/article/details/19204529 兩個棧實現一個隊列&#xff1a; 原理方法&#xff1a;用一個棧為主棧&#xff0c;一個棧為輔助棧存放臨時元素。 入隊&#xff1a;將元素依次壓入主棧 出隊&#xff1a;先檢測輔助棧是否為空&#xff0c;如果非空&a…

UVa11426——歐拉函數

發現對于gcd問題要多和歐拉函數聯系在一起&#xff0c;雖然有時候并不是互質&#xff0c;但是我們知道有多少互質的然后根據互質的數目就能解決很多個gcd的問題 對于這道題目&#xff0c;題目要求的是所有數對的gcd的和&#xff0c;直接思考的話有難度。但是我們如果聯想到歐拉…

C++:繼承和多態

虛函數:只有類的成員函數才能定義為虛函數 虛函數 在類的成員函數前面加上一個 virtual 關鍵字, 此時這個成員函數就叫做虛函數 虛函數 當在子類中定義了一個與父類完全相同的虛函數的時候,此時就叫做子類的虛函數重寫了父類的虛函數 構成多態的條件 派生類重寫基類的虛函數…

POJ 1061擴展歐幾里得

擴展歐幾里得的模板題&#xff0c;需要注意的是為了得到一個最小正數解我們要使axbyc中的a,b都是正數 #include<cstdio> #include<cstring> #include<cstdlib> #include<algorithm> #include<iostream> #include<cmath> #include<ctim…