C、C++之動(dòng)態(tài)數(shù)組的實(shí)現(xiàn)二(C++版本)

c、c++動(dòng)態(tài)數(shù)組(c++版本)

本篇文章基于筆者正在參與的c++課程,第二次作業(yè)的內(nèi)容是要求使用c++的特性對(duì)上一次的程序?qū)崿F(xiàn)改進(jìn)并封裝(上一版本戳我)。

嚴(yán)格來(lái)說(shuō),上一個(gè)版本不能算是純粹的C語(yǔ)言版本,這是因?yàn)榇a中使用了c++的引用特性,這是C語(yǔ)言所不包含的。然而,這是由于測(cè)試代碼的限制,因而我們還是把它看做C語(yǔ)言的實(shí)現(xiàn)。(也可以編寫一種不包含引用的代碼來(lái)達(dá)到相同的效果,這要求使用到宏定義和一種稱之為“wrapper”的小技巧)

閑話少敘,先放出新的測(cè)試代碼,再具體討論各個(gè)函數(shù)的改寫方法。

//LibArray.cpp
// 實(shí)驗(yàn)內(nèi)容:
// 1:將C語(yǔ)言版本LibArray用C++封裝,注意,原C版本保留一個(gè)備份

// 實(shí)驗(yàn)?zāi)康模?// 1:C++類定義的基本方法

// 只提交CLibArray.cpp及CLibArray.h

#include "stdafx.h"
#include <assert.h>
#include "CLibArray.h"

int _tmain(int argc, _TCHAR* argv[])
{
    CArray array;
    // 不再需要initial,但應(yīng)該有正確的初始化
    // array_initial(array); 

    //array.recap(10); 
    //assert(array.capacity() == 10); 

    //////////////////////////////////////////////////////////////////////////
    for (int i = 0; i < 20; ++i)
    {
        array.append(i); 
    }
    assert(array.size() == 20); 

    for (int i = 0; i < array.size(); ++i)
    {
        assert(array.at(i) == i); 
    }

    //////////////////////////////////////////////////////////////////////////
    CArray array2, array3; 
    // array_initial(array2); 
    // array_initial(array3); 

    array2.copy(array); 
    assert(array.compare(array2) == true); 

    array3.copy(array); 
    assert(array.compare(array3) == true); 

    //////////////////////////////////////////////////////////////////////////
    array2.insert(2, 3); 
    assert(array.compare(array2) == false); 

    //////////////////////////////////////////////////////////////////////////
    array3.at(2) = 5; 
    assert(array.compare(array3) == false); 

    //////////////////////////////////////////////////////////////////////////
    // 不再需要destroy,但應(yīng)該有正確的內(nèi)存釋放
    // array_destroy(array); 
    // array_destroy(array2); 
    // array_destroy(array3); 

    return 0;
}

類的定義

從測(cè)試代碼和注釋中可見(jiàn),對(duì)變量 array,我們只需要定義一個(gè)合適的 CArray 類即可。而在類的定義部分,將原有的結(jié)構(gòu)體成員放到 private 私有數(shù)據(jù)成員:


typedef int TypeName;
const int INITLENGTH = 10; //初始化長(zhǎng)度可以為任意正整數(shù),也可以為零,但需要把a(bǔ)ppend函數(shù)里的語(yǔ)句做適當(dāng)修改

private:
    TypeName *arrayhead;
    int arraysize;
    int arraycapacity;

自然而然地,我們需要將C語(yǔ)言版的各個(gè)函數(shù),放到 CArray 類的公有部分作為接口。
另一方面,注意到注釋部分:不再需要 initial 和 destroy 函數(shù),學(xué)過(guò)c++的盆友們都知道,這是很自然的,因?yàn)樵谑褂胏++編程時(shí),一般需要給自己定義的類寫好對(duì)應(yīng)的構(gòu)造函數(shù)和析構(gòu)函數(shù),實(shí)際上這樣的兩種函數(shù),就對(duì)應(yīng)于原來(lái)的 initial 和 destroy 函數(shù)。(關(guān)于構(gòu)造函數(shù)和析構(gòu)函數(shù),可以參考這篇文章
另外,注意到測(cè)試文件不再包含下列語(yǔ)句 array.recap(10);由于在C語(yǔ)言實(shí)現(xiàn)中, recap 函數(shù)用于給動(dòng)態(tài)數(shù)組賦予一定的空間大小, 而測(cè)試函數(shù)中取消了此語(yǔ)句,那么就有兩種可能的操作,一種是要考慮將空間的分配放在其他的函數(shù)中,或者保留 recap 函數(shù),再讓其他函數(shù)調(diào)用它。這樣的特性很符合類的 protected 方法的定義。(當(dāng)然,若不考慮繼承,將其作為 private 方法也未嘗不可)
于是總體思路清晰了:在 private 成員中定義了動(dòng)態(tài)數(shù)組的必要參數(shù),在 public 部分定義了可以進(jìn)行的操作:

public:
    CArray();
    ~CArray();
    inline int capacity() { return arraycapacity; };
    inline int size() { return arraysize; };
    inline TypeName& at(int num) { return arrayhead[num]; };
    void append(int num);
    void copy(CArray &another);
    bool compare(CArray &another);
    void insert(int num, TypeName value);

protected:
    void recap(int length);
    void printarray();

簡(jiǎn)短的函數(shù)直接定義為內(nèi)聯(lián)函數(shù)(注意若直接寫在類定義的頭文件內(nèi),則無(wú)需 inline 關(guān)鍵字,然而是否最終編譯為內(nèi)聯(lián)函數(shù),取決于編譯器的具體實(shí)現(xiàn),關(guān)于 inline 關(guān)鍵字,可以參考這篇博客

函數(shù) printarray 用來(lái)輸出動(dòng)態(tài)數(shù)組的關(guān)鍵信息。

在改寫函數(shù)時(shí),主要工作是修改其參數(shù),并在具體的定義中省去對(duì)調(diào)用對(duì)象本身的顯式表示(也可以采用 this 指針來(lái)完成)。

構(gòu)造函數(shù)和析構(gòu)函數(shù)

在構(gòu)造函數(shù)中,可以選擇給頭指針?lè)峙湟欢ù笮〉膬?nèi)存,也可以賦值為空(nullptr,即C語(yǔ)言中的NULL),出于一種合情合理的原因,我給它分配了一定的大小。
在 c++中,使用 new 和 delete 來(lái)分配和釋放內(nèi)存,

CArray::CArray()
{
    arrayhead = new TypeName[INITLENGTH];
    arraysize = 0;
    arraycapacity = INITLENGTH;
}

而對(duì)于析構(gòu)函數(shù),只要相應(yīng)地釋放內(nèi)存即可:

CArray::~CArray()
{
    delete[] arrayhead;
    arrayhead = nullptr;
    arraycapacity = 0;
    arraysize = 0;
}

注意:釋放內(nèi)存是必需步驟。且 delete[]new []相對(duì)應(yīng)。(關(guān)于 new 和 delete 的注意事項(xiàng)及原理, 可參閱這篇博客

recap函數(shù)

此函數(shù)是內(nèi)存分配的關(guān)鍵,首先需保證其參數(shù)(capacity)合法,接下來(lái),先新定義一個(gè)同類型的指針來(lái)指向需要的內(nèi)存大小(capacity)的地址,并把原有的 arrayhead 所指向的內(nèi)存里的數(shù)據(jù)復(fù)制到新的內(nèi)存中,再刪去原有的數(shù)據(jù), 并讓 arrayhead 指向新的內(nèi)存地址。
這樣的做的原因是,由于參數(shù) capacity 與原有數(shù)組的 arraycapacity 和 arraysize 的大小關(guān)系不確定,因此只能新分配一塊內(nèi)存,再將所需內(nèi)存大小的數(shù)據(jù)進(jìn)行轉(zhuǎn)存。
最后需要修改相關(guān)的 private 成員的值。

void  CArray::recap(int capacity)
{
    if (capacity < 0)
    {
        cout << "array's length should be larger than zero, check out array_recap()" << endl;
        exit(EXIT_FAILURE);
    }

    arraycapacity = capacity;
    arraysize = arraysize > capacity ? capacity : arraysize;

    TypeName* buffer = nullptr;
    buffer = new TypeName[capacity];
    memcpy(buffer, arrayhead, sizeof(TypeName) * arraycapacity);

    delete[] arrayhead;
    arrayhead = buffer;

    //分配失敗
    if (arrayhead == nullptr)
    {
        cout << "malloc failed in array_recap()." << endl;
        exit(0);
    }
}

append 函數(shù)

該函數(shù)所需要做的工作是給第 num 位的成員賦值為 num。 為了達(dá)到此目的,需要檢查 num 和 capacity 之間的關(guān)系,若 num > arraycapacity,則需要使用 recap 函數(shù)擴(kuò)大數(shù)組的容量,擴(kuò)大的方式和大小可自行定義, 在此采取每次擴(kuò)大一倍的方式,這樣,算法的復(fù)雜度將由O(n) 減小為 O(c),此原理不詳述。
注意到每次擴(kuò)大一倍容量的前提是, capacity不為零,因此,在構(gòu)造函數(shù)中, 我選擇給數(shù)組一個(gè)不為零的內(nèi)存大小,當(dāng)然,如果堅(jiān)持在構(gòu)造函數(shù)中要使用 arrayhead = nullptr;那么在 append 函數(shù)中, 可以使用 (*this).recap((arraysize +1)*2); 這樣的代碼。

//給 arrayhead 數(shù)組的第 num 位賦值為 num, 若 num 大于實(shí)際長(zhǎng)度,則擴(kuò)充長(zhǎng)度至 num
void  CArray::append(int num)
{
    if (num + 1 > arraycapacity)
    {
        // 一次擴(kuò)大為原來(lái)的兩倍, 時(shí)間復(fù)雜度更小(O(n) - > O(c))
        (*this).recap(arraysize *2);
    }
    arrayhead[arraysize++] = num;
}

copy函數(shù)

和C語(yǔ)言版本基本一致,我們需要做的工作是使調(diào)用對(duì)象的容量與被復(fù)制的對(duì)象大小一致,然后將所有數(shù)據(jù)復(fù)制到調(diào)用對(duì)象的內(nèi)存中,這樣可以實(shí)現(xiàn)動(dòng)態(tài)數(shù)組的復(fù)制。

//將 another 對(duì)象復(fù)制給調(diào)用的對(duì)象
void  CArray::copy(CArray &another)
{
    (*this).recap(another.capacity());
    memcpy(arrayhead, another.arrayhead, another.size() * sizeof(TypeName));
    arraysize = another.size();
}

注意:使用 memcpy 函數(shù),比用 for 循環(huán)逐一賦值的效率要高,因?yàn)?mencpy 可以充分利用數(shù)據(jù)總線的位數(shù)進(jìn)行傳輸。

compare函數(shù)

此函數(shù)幾乎沒(méi)有要修改的地方,唯一值得注意的是,為了輸出比較的結(jié)果,我選擇了用 for 循環(huán)來(lái)逐一比較數(shù)據(jù), 若只要求其總體比較的結(jié)果,可以使用 memcmp 函數(shù),效率更高,也更簡(jiǎn)潔。

bool  CArray::compare(CArray &another)
{
    //another.printarray();
    //輸出 size 不同的信息
    if (another.size() != arraysize)
    {
        cout << "Their size are not equal, check out in CArray::compare()." << endl;
        return false;
    }

    if (another.capacity() != arraycapacity)
    {
        cout << "Their capacity are not equal, check out in CArray::compare()." << endl;
        return false;
    }
    //為了輸出是第幾位不同,采用循環(huán),否則可以采用以下語(yǔ)句,效率更高
    //return memcmp(another.arrayhead, arrayhead, size() * sizeof(TypeName)) == 0;
    for (int i = 0; i < another.size(); i++)
        if (another.arrayhead[i] != arrayhead[i])
        {
            cout << "They are not equal in the NO." << i << " place" << endl;
            return false;
        }

    return true;
}

insert 函數(shù)

此函數(shù)需要在 第 index 的位置插入一個(gè)值為 value 的元素,為此,只需要使用 recap 函數(shù)每次多增加一個(gè)單位長(zhǎng)度的內(nèi)存空間即可,順次移動(dòng) index 前的各位,然后將第 index 位賦值。

void  CArray::insert(int index, TypeName value)
{
    if (index > arraycapacity || index < 0)
    {
        cout << "Cannot insert with an invaild index, check out in array_insert()." << endl;
        exit(EXIT_FAILURE);
    }
    else
    {
        (*this).recap(arraycapacity + 1);

        for (int i = arraycapacity - 1; i > index; i--)
            arrayhead[i] = arrayhead[i - 1];
        arrayhead[index] = value;

        arraysize ++;
    }
}

printarray函數(shù)

此函數(shù)的實(shí)現(xiàn)因人而異,只需要獲得調(diào)試時(shí)所需要的信息即可。

總結(jié)

本篇博客實(shí)現(xiàn)了動(dòng)態(tài)數(shù)組的C++版本的一種簡(jiǎn)單實(shí)現(xiàn)。未完善的地方必然存在,望廣大讀者批評(píng)指正。為了符合測(cè)試程序, 我沒(méi)有采用模板進(jìn)行代碼的編寫,雖然在上一版的實(shí)驗(yàn)要求中提到了,對(duì)程序的要求是能夠各種不同類型的數(shù)據(jù)。然而代碼中沒(méi)有使用模板進(jìn)行對(duì)象的聲明,可知無(wú)需進(jìn)行模板類和模板函數(shù)的編寫。
關(guān)于異常的編寫,代碼中某些函數(shù)使用了諸如exit(EXIT_FAILURE)這樣的語(yǔ)句。在 C++ 中,更為規(guī)范的實(shí)現(xiàn)是拋出異常, 再進(jìn)行針對(duì)性的處理,由于測(cè)試程序較為簡(jiǎn)單,因此沒(méi)有編寫針對(duì)異常的代碼

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,963評(píng)論 6 542
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,348評(píng)論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 178,083評(píng)論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 63,706評(píng)論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,442評(píng)論 6 412
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 55,802評(píng)論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,795評(píng)論 3 446
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 42,983評(píng)論 0 290
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,542評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,287評(píng)論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,486評(píng)論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,030評(píng)論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,710評(píng)論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 35,116評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 36,412評(píng)論 1 294
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,224評(píng)論 3 398
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,462評(píng)論 2 378

推薦閱讀更多精彩內(nèi)容

  • C、C++之動(dòng)態(tài)數(shù)組的實(shí)現(xiàn) 本篇博客基于筆者本人正在學(xué)習(xí)的C++上機(jī)課程作業(yè),主要代碼由C語(yǔ)言構(gòu)成。由于C語(yǔ)言沒(méi)有...
    largerthanlife閱讀 1,281評(píng)論 0 1
  • 第5章 引用類型(返回首頁(yè)) 本章內(nèi)容 使用對(duì)象 創(chuàng)建并操作數(shù)組 理解基本的JavaScript類型 使用基本類型...
    大學(xué)一百閱讀 3,263評(píng)論 0 4
  • 一、溫故而知新 1. 內(nèi)存不夠怎么辦 內(nèi)存簡(jiǎn)單分配策略的問(wèn)題地址空間不隔離內(nèi)存使用效率低程序運(yùn)行的地址不確定 關(guān)于...
    SeanCST閱讀 7,861評(píng)論 0 27
  • 感賞想兒早起上學(xué)。 感賞老公在工作中經(jīng)驗(yàn)豐富,客人在電話里那么一說(shuō),老公就明白了客戶的需求,把貨帶到了現(xiàn)場(chǎng)一裝,對(duì)...
    清晨劉丹閱讀 192評(píng)論 3 3
  • 目錄 前情回顧:冷杉 次日醒來(lái)推開(kāi)窗的時(shí)候,一陣夾雜著水霧的冷氣猝不及防地闖入屋里,和呼吸進(jìn)胸腔中的寒氣一內(nèi)一外地...
    原小尚閱讀 561評(píng)論 2 4