您好,登錄后才能下訂單哦!
在C++中有C沒有的string字符串類型,string類型的數(shù)據(jù)其實是一個指向字符串首地址的指針變量,因此在string類的默認(rèn)成員函數(shù)拷貝構(gòu)造和賦值運算符的重載就會涉及到深淺拷貝的問題,一不小心要么就是內(nèi)存泄露要么就是多次釋放同一塊空間導(dǎo)致程序崩潰,下面就來模擬實現(xiàn)一個簡潔版的String類:
既然是指向一個字符串的指針,因此類的成員變量就需要有一個char*類型的指針;
#include <iostream> #include <string.h> using namespace std; class CMyString { public: CMyString(const char* str); CMyString(const CMyString& s); CMyString& operator=(CMyString s); //CMyString& operator=(const CMyString& s); ~CMyString(); void print_string(); private: char* _str; };
上面為簡單的CMyString類的聲明,接下來要實現(xiàn)最主要的四個默認(rèn)的成員函數(shù):
構(gòu)造函數(shù)主要是為成員變量_str分配內(nèi)存空間并且初始化為形參的值;
CMyString::CMyString(const char* str) //含參的構(gòu)造函數(shù) :_str(NULL) { assert(str); _str = new char[_capacity]; strcpy(_str, str); } CMyString::CMyString() //默認(rèn)的構(gòu)造函數(shù) :_str(NULL) {}
拷貝構(gòu)造就會涉及到了深淺拷貝的問題,因為不能使兩個字符串的指針指向同一塊地址空間:
CMyString::CMyString(const CMyString& s) :_str(NULL) { CMyString tmp(s._str);//用前面實現(xiàn)的構(gòu)造函數(shù)構(gòu)造出一個值為s._str的臨時類對象 swap(_str, tmp._str);//交換臨時類的字符串和_str,這樣當(dāng)tmp出了作用域就會自動釋放 }
賦值運算符的重載函數(shù)同樣會涉及到深淺拷貝的問題:
//這是一種比較現(xiàn)代的寫法,沒有用引用s就為一個臨時的類對象,出了作用域就會自動調(diào)用析構(gòu)函數(shù) CMyString& CMyString::operator=(CMyString s) { if(strcmp(s._str, _str) != 0) swap(_str, s._str);//交換二者的值就能將有效值賦給_str,而原來的值隨s釋放 return *this; } //較為傳統(tǒng)的寫法,先要判斷是否自己給自己賦值再釋放自己空間,重新開辟一塊空間拷貝所需值 //CMyString& CMyString::operator=(const CMyString& s) //{ // if(this != &s) // { // delete[] _str; // _str = new char[strlen(s.str)+1]; // strcpy(_str, s._str); // } // return *this; //}
但是在上面注釋掉的一種寫法中存在一個問題,就是如果將自己本身的地址空間釋放掉了之后,再去new一塊空間有可能會new不出來,這樣的話不僅不能成功賦值,連自身本就存在的值也丟掉了,因此可以優(yōu)化為如下代碼:
CMyString& CMyString::operator=(const CMyString& s) { if(this != &s) { char *tmp = new char[strlen(s.str)+1]; if(tmp != NULL) { delete[] _str; _str = tmp; strcpy(_str, s.str); } } return *this; }
析構(gòu)函數(shù)是在當(dāng)類對象出了所在作用域時自動調(diào)用完成清理工作的:
CMyString::~CMyString() { if(_str != NULL) //檢查類成員是否為空,delete不能釋放空指針 { delete[] _str; _str = NULL; //防止出現(xiàn)野指針 } }
最后一個print_string函數(shù)是為了打印驗證結(jié)果,這里就不寫了;
main函數(shù):
int main() { CMyString s1("this is my string..."); s1.print_string(); CMyString s2("hello world..."); s2.print_string(); CMyString s3(s2); s3.print_string(); s3 = s1; s3.print_string(); return 0; }
運行程序結(jié)果如下:
《完》
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。