編寫(xiě)一個(gè)帶指針成員變量的類(lèi)
1. 拷貝函數(shù)
- 當(dāng)類(lèi)中有指針成員變量時(shí)宰缤,必須要實(shí)現(xiàn)拷貝函數(shù)颂翼,不能使用系統(tǒng)提供的默認(rèn)方法晃洒。
2. Big Three
- 拷貝構(gòu)造
- 拷貝賦值
- 析構(gòu)函數(shù)
字符串的兩種存儲(chǔ)形式:最前存貯字符串的長(zhǎng)度慨灭;最后有結(jié)束符'\0'
2.1 拷貝構(gòu)造函數(shù)
- 構(gòu)造函數(shù)的參數(shù)類(lèi)型為本class
- new一份同樣大小的內(nèi)存,再使用strcpy
大小要+1球及,為結(jié)束位的'\0'留出位置
2.1 拷貝賦值函數(shù)
- 重載運(yùn)算符=
- 監(jiān)測(cè)是否自我賦值氧骤。之后先清空,在申請(qǐng)同樣大小的內(nèi)存吃引,再拷貝strcopy
如果沒(méi)有做自我賦值檢測(cè)筹陵,會(huì)出現(xiàn)把自己清空后再賦值,這時(shí)調(diào)用指針會(huì)出現(xiàn)不確定結(jié)果
3. 堆棧
- 堆镊尺,heap朦佩。由操作系統(tǒng)提供的一塊global內(nèi)存空間,程序可動(dòng)態(tài)分配庐氮。通過(guò) new去獲得语稠。malloc
- 局部的指針變量如果沒(méi)有被delete,那么指針變量的堆空間將無(wú)法被安全釋放弄砍,造成了內(nèi)存泄漏仙畦。
- new的流程:先分配memory,再調(diào)用構(gòu)造函數(shù)音婶。
void* mem = operator new( sizeof(Complex) ); //分配內(nèi)存慨畸,調(diào)用malloc(sizeof())
pc = static_cast<Complex*>(mem); //類(lèi)型轉(zhuǎn)換
pc->Complex::Complex(1,2); //構(gòu)造函數(shù)
- 堆,heap
- delete的流程:先調(diào)用析構(gòu)函數(shù)衣式,再釋放memory
String::~String(ps); //析構(gòu)函數(shù)
operator delete(ps); //釋放內(nèi)存寸士,內(nèi)部調(diào)用free
- 棧,stack碴卧。存在于作用域內(nèi)的一塊內(nèi)存空間弱卡。
- 局部對(duì)象,會(huì)被自動(dòng)清理螟深。
- static關(guān)鍵字谐宙。靜態(tài)對(duì)象的生命周期和程序綁定,和作用域無(wú)關(guān)界弧。
- 全局對(duì)象凡蜻。
4. 內(nèi)存中new時(shí)的真正形態(tài)
4.1 單個(gè)對(duì)象時(shí)
- debug模式
- 每申請(qǐng)一個(gè)對(duì)象搭综,內(nèi)存中的實(shí)際大小為:
- 對(duì)象本身的內(nèi)存大小
- 包裹對(duì)象的內(nèi)存32字節(jié)+4字節(jié)
- cookie內(nèi)存4+4,在整個(gè)內(nèi)存塊的頭和尾
- 全部大小相加后划栓,再向16的倍數(shù)對(duì)齊
- 每申請(qǐng)一個(gè)對(duì)象搭综,內(nèi)存中的實(shí)際大小為:
- release模式
- 每申請(qǐng)一個(gè)對(duì)象兑巾,內(nèi)存中的實(shí)際大小為:
- 對(duì)象本身的內(nèi)存大小
- cookie內(nèi)存4+4,在整個(gè)內(nèi)存塊的頭和尾
- 全部大小相加后忠荞,再向16的倍數(shù)對(duì)齊
- 每申請(qǐng)一個(gè)對(duì)象兑巾,內(nèi)存中的實(shí)際大小為:
- cookie內(nèi)存的作用
- 標(biāo)記回收時(shí)的內(nèi)存大小
- 因?yàn)橐欢ㄊ?6的倍數(shù)蒋歌,所以最后一位一定是0,可以利用最后一位的0/1來(lái)標(biāo)示當(dāng)前內(nèi)存塊是給出狀態(tài)還是回收狀態(tài)
4.2 數(shù)組對(duì)象時(shí)
- new type[] 要搭配 delete []
- debug模式
- 每申請(qǐng)一個(gè)對(duì)象委煤,內(nèi)存中的實(shí)際大小為:
- 記錄數(shù)組大刑糜汀:4字節(jié)
- 數(shù)組長(zhǎng)度個(gè)的對(duì)象本身的內(nèi)存大小:n * object
- 包裹整個(gè)數(shù)組和大小的內(nèi)存32字節(jié)+4字節(jié)
- cookie內(nèi)存4+4碧绞,在整個(gè)內(nèi)存塊的頭和尾
- 全部大小相加后府框,再向16的倍數(shù)對(duì)齊
- 每申請(qǐng)一個(gè)對(duì)象委煤,內(nèi)存中的實(shí)際大小為:
- release模式
- 每申請(qǐng)一個(gè)對(duì)象,內(nèi)存中的實(shí)際大小為:
- 記錄數(shù)組大屑チ凇:4字節(jié)
- 數(shù)組長(zhǎng)度個(gè)的對(duì)象本身的內(nèi)存大衅染浮:n * object
- cookie內(nèi)存4+4,在整個(gè)內(nèi)存塊的頭和尾
- 全部大小相加后兴使,再向16的倍數(shù)對(duì)齊
- 每申請(qǐng)一個(gè)對(duì)象,內(nèi)存中的實(shí)際大小為:
5. 類(lèi)模版
template<typename T>
class complex
{
public:
complex (T r = 0, T i = 0)
: re (r), im (i)
{}
private:
T re, im;
}
6. 函數(shù)模版
template <class T>
inline
const T& min(const T& a, const T& b)
{
return b < a ? b : a;
}
思考
1. new type[] 和 delete [] 成對(duì)出現(xiàn)的原因系宜?
- 如果不成對(duì)出現(xiàn),會(huì)造成內(nèi)存泄漏发魄,但是泄漏的內(nèi)存并不是數(shù)組中對(duì)象的內(nèi)存盹牧,數(shù)組中的所有對(duì)象已經(jīng)被正確釋放。delete[]的作用是欠母,通知編譯器釋放的是個(gè)數(shù)組對(duì)象欢策,然后調(diào)用多次的析構(gòu)函數(shù);如果只是調(diào)用delete赏淌,只會(huì)調(diào)用一次析構(gòu)函數(shù)踩寇。
- 所以真正內(nèi)存泄漏的內(nèi)存,是數(shù)組對(duì)象中指針?biāo)赶虻哪切﹥?nèi)存六水。也就是說(shuō)如果當(dāng)數(shù)組對(duì)象的內(nèi)部變量沒(méi)有指針時(shí)俺孙,是不會(huì)出現(xiàn)問(wèn)題。
- 要保持良好的變成習(xí)慣掷贾,array new 和 array delete要成對(duì)出現(xiàn)睛榄。
2. ' * '和' & '標(biāo)志符在不同位置時(shí)的不同含義
- ' & '在類(lèi)型名后' typename& ',表示為引用想帅;' & '在對(duì)象前' &object '场靴,表示為取地址
3. 回答第一周筆記中的第一個(gè)思考問(wèn)題,為什么可以數(shù)據(jù)需要單獨(dú)存儲(chǔ),而函數(shù)可以公用一份?
- 通過(guò)對(duì)象調(diào)用類(lèi)方法時(shí)的真相
complex c1,c2;
cout << c1.real(); // 實(shí)際情況旨剥,cout << complex::real(&c1);
cout << c2.real(); // 實(shí)際情況咧欣,cout << complex::real(&c2);
4. 為什么靜態(tài)函數(shù)只能使用靜態(tài)數(shù)據(jù),而不能使用直接使用類(lèi)的私有數(shù)據(jù)轨帜?
- 因?yàn)閟tatic函數(shù)的參數(shù)列表中魄咕,沒(méi)有默認(rèn)的this指針。
- 靜態(tài)函數(shù)可以通過(guò)類(lèi)名調(diào)用蚌父,也可以使用對(duì)象調(diào)用哮兰。
5. 類(lèi)模版和函數(shù)模版的使用區(qū)別
- 類(lèi)模版使用時(shí)必須明確指出模版的類(lèi)型;函數(shù)模版使用時(shí)苟弛,不用明確指出喝滞,編譯器會(huì)進(jìn)行argument deduction,參數(shù)推導(dǎo)