溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

自我實現(xiàn)一個簡潔版的String類——1

發(fā)布時間:2020-07-22 23:08:05 來源:網(wǎng)絡(luò) 閱讀:439 作者:給我個bit位 欄目:編程語言

    在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é)果如下:

自我實現(xiàn)一個簡潔版的String類——1



《完》

向AI問一下細(xì)節(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)容。

AI