C、C++之动态数组的实现二(C++版本)

c、c++动态数组(c++版本)

本篇文章基于笔者正在参与的c++课程,第二次作业的内容是要求使用c++的特性对上一次的程序实现改进并封装(上一版本戳我)。

严格来说,上一个版本不能算是纯粹的C语言版本,这是因为代码中使用了c++的引用特性,这是C语言所不包含的。然而,这是由于测试代码的限制,因而我们还是把它看做C语言的实现。(也可以编写一种不包含引用的代码来达到相同的效果,这要求使用到宏定义和一种称之为“wrapper”的小技巧)

闲话少叙,先放出新的测试代码,再具体讨论各个函数的改写方法。

//LibArray.cpp
// 实验内容:
// 1:将C语言版本LibArray用C++封装,注意,原C版本保留一个备份

// 实验目的:
// 1:C++类定义的基本方法

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

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

int _tmain(int argc, _TCHAR* argv[])
{
    CArray array;
    // 不再需要initial,但应该有正确的初始化
    // 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,但应该有正确的内存释放
    // array_destroy(array); 
    // array_destroy(array2); 
    // array_destroy(array3); 

    return 0;
}

类的定义

从测试代码和注释中可见,对变量 array,我们只需要定义一个合适的 CArray 类即可。而在类的定义部分,将原有的结构体成员放到 private 私有数据成员:


typedef int TypeName;
const int INITLENGTH = 10; //初始化长度可以为任意正整数,也可以为零,但需要把append函数里的语句做适当修改

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

自然而然地,我们需要将C语言版的各个函数,放到 CArray 类的公有部分作为接口。
另一方面,注意到注释部分:不再需要 initial 和 destroy 函数,学过c++的盆友们都知道,这是很自然的,因为在使用c++编程时,一般需要给自己定义的类写好对应的构造函数和析构函数,实际上这样的两种函数,就对应于原来的 initial 和 destroy 函数。(关于构造函数和析构函数,可以参考这篇文章
另外,注意到测试文件不再包含下列语句 array.recap(10);由于在C语言实现中, recap 函数用于给动态数组赋予一定的空间大小, 而测试函数中取消了此语句,那么就有两种可能的操作,一种是要考虑将空间的分配放在其他的函数中,或者保留 recap 函数,再让其他函数调用它。这样的特性很符合类的 protected 方法的定义。(当然,若不考虑继承,将其作为 private 方法也未尝不可)
于是总体思路清晰了:在 private 成员中定义了动态数组的必要参数,在 public 部分定义了可以进行的操作:

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();

简短的函数直接定义为内联函数(注意若直接写在类定义的头文件内,则无需 inline 关键字,然而是否最终编译为内联函数,取决于编译器的具体实现,关于 inline 关键字,可以参考这篇博客

函数 printarray 用来输出动态数组的关键信息。

在改写函数时,主要工作是修改其参数,并在具体的定义中省去对调用对象本身的显式表示(也可以采用 this 指针来完成)。

构造函数和析构函数

在构造函数中,可以选择给头指针分配一定大小的内存,也可以赋值为空(nullptr,即C语言中的NULL),出于一种合情合理的原因,我给它分配了一定的大小。
在 c++中,使用 new 和 delete 来分配和释放内存,

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

而对于析构函数,只要相应地释放内存即可:

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

注意:释放内存是必需步骤。且 delete[]new []相对应。(关于 new 和 delete 的注意事项及原理, 可参阅这篇博客

recap函数

此函数是内存分配的关键,首先需保证其参数(capacity)合法,接下来,先新定义一个同类型的指针来指向需要的内存大小(capacity)的地址,并把原有的 arrayhead 所指向的内存里的数据复制到新的内存中,再删去原有的数据, 并让 arrayhead 指向新的内存地址。
这样的做的原因是,由于参数 capacity 与原有数组的 arraycapacity 和 arraysize 的大小关系不确定,因此只能新分配一块内存,再将所需内存大小的数据进行转存。
最后需要修改相关的 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 函数

该函数所需要做的工作是给第 num 位的成员赋值为 num。 为了达到此目的,需要检查 num 和 capacity 之间的关系,若 num > arraycapacity,则需要使用 recap 函数扩大数组的容量,扩大的方式和大小可自行定义, 在此采取每次扩大一倍的方式,这样,算法的复杂度将由O(n) 减小为 O(c),此原理不详述。
注意到每次扩大一倍容量的前提是, capacity不为零,因此,在构造函数中, 我选择给数组一个不为零的内存大小,当然,如果坚持在构造函数中要使用 arrayhead = nullptr;那么在 append 函数中, 可以使用 (*this).recap((arraysize +1)*2); 这样的代码。

//给 arrayhead 数组的第 num 位赋值为 num, 若 num 大于实际长度,则扩充长度至 num
void  CArray::append(int num)
{
    if (num + 1 > arraycapacity)
    {
        // 一次扩大为原来的两倍, 时间复杂度更小(O(n) - > O(c))
        (*this).recap(arraysize *2);
    }
    arrayhead[arraysize++] = num;
}

copy函数

和C语言版本基本一致,我们需要做的工作是使调用对象的容量与被复制的对象大小一致,然后将所有数据复制到调用对象的内存中,这样可以实现动态数组的复制。

//将 another 对象复制给调用的对象
void  CArray::copy(CArray &another)
{
    (*this).recap(another.capacity());
    memcpy(arrayhead, another.arrayhead, another.size() * sizeof(TypeName));
    arraysize = another.size();
}

注意:使用 memcpy 函数,比用 for 循环逐一赋值的效率要高,因为 mencpy 可以充分利用数据总线的位数进行传输。

compare函数

此函数几乎没有要修改的地方,唯一值得注意的是,为了输出比较的结果,我选择了用 for 循环来逐一比较数据, 若只要求其总体比较的结果,可以使用 memcmp 函数,效率更高,也更简洁。

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;
    }
    //为了输出是第几位不同,采用循环,否则可以采用以下语句,效率更高
    //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 函数

此函数需要在 第 index 的位置插入一个值为 value 的元素,为此,只需要使用 recap 函数每次多增加一个单位长度的内存空间即可,顺次移动 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函数

此函数的实现因人而异,只需要获得调试时所需要的信息即可。

总结

本篇博客实现了动态数组的C++版本的一种简单实现。未完善的地方必然存在,望广大读者批评指正。为了符合测试程序, 我没有采用模板进行代码的编写,虽然在上一版的实验要求中提到了,对程序的要求是能够各种不同类型的数据。然而代码中没有使用模板进行对象的声明,可知无需进行模板类和模板函数的编写。
关于异常的编写,代码中某些函数使用了诸如exit(EXIT_FAILURE)这样的语句。在 C++ 中,更为规范的实现是抛出异常, 再进行针对性的处理,由于测试程序较为简单,因此没有编写针对异常的代码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容

  • C、C++之动态数组的实现 本篇博客基于笔者本人正在学习的C++上机课程作业,主要代码由C语言构成。由于C语言没有...
    largerthanlife阅读 1,258评论 0 1
  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,219评论 0 4
  • 一、温故而知新 1. 内存不够怎么办 内存简单分配策略的问题地址空间不隔离内存使用效率低程序运行的地址不确定 关于...
    SeanCST阅读 7,787评论 0 27
  • 感赏想儿早起上学。 感赏老公在工作中经验丰富,客人在电话里那么一说,老公就明白了客户的需求,把货带到了现场一装,对...
    清晨刘丹阅读 183评论 3 3
  • 目录 前情回顾:冷杉 次日醒来推开窗的时候,一阵夹杂着水雾的冷气猝不及防地闯入屋里,和呼吸进胸腔中的寒气一内一外地...
    原小尚阅读 540评论 2 4