当前位置: 首页 > article >正文

典型数据结构的模板实现

栈和数组

  • 1.使用类模板实现数组结构
    • 定长数组
    • 可变数组
  • 2.使用类模板实现栈结构

在我们初步了解编写模板类后,应当做一下代码练习。这节我们就做一个编写代码的补充,方便大家继续学习模板类的嵌套。作为新手而言,建议大家先写一个具体类,调试好后再去修改成模板类,因为调试模板类会相对复杂。

1.使用类模板实现数组结构

数组是我们常用的一种数据类型,我们今天的内容就先从编写一个数组类模板开始。数组有定长数组和边长数组的区别,我们只来实现它们的部分功能。

定长数组

首先我们先学习编写一个int类型的定长数组数组类。我们只实现查看和改写数组的元素即可。这个过程中涉及到重载[]运算符:

#define Arraysize 10
class Array
{
    private:
        int items[Arraysize]; // 声明一个数组,这里的[]不是运算符
    public:
        int sign=0; // 用于标记重载的[]运算符使用了几次
        Array(){memset(items,0,sizeof(items));} // 使用memset函数初始化数组,需要知道首地址、初始化值、数组元素类型所占空间
        int& operator[] (int ii) // 重载[]运算符,返回值类型为int的引用
        {
            this->sign++; // this指针类似于python中的self,他默认标记类地址,可以查找类内容
            return *(items+ii);
        }
        const int& operator[] (int ii) // 如果是const修饰的数组,上面的函数将不会被调用
        const{
            return *(items+ii);
        }
};

这段代码里我偷偷用了一些没有讲到的函数和知识点。包括重载运算符,初始化函数和类的this指针。当然这段代码当然没有使用this指针的必要,this->sign++直接写成sign++效果也是一样的。
写好之后我们来测试一下:

int main()
{
    Array a;
    a[0]=1;a[1]=2;a[2]=3;a[3]=4;
    for(int i=0;i<5;i++)
    {
        cout<<a[i]<<" ";
    }
    cout<<"\n"<<a.sign<<endl;
}
// 输出为:1 2 3 4 0 
//		  9

接下来我们把它改成函数模板的样子。因为函数模板可以定义已知类型,所以我们也可以这样写:

template <class T, int Arraysize=10>
class Array
{
    private:
        T items[Arraysize]; // 声明一个数组,这里的[]不是运算符
    public:
        int sign=0; // 用于标记重载的[]运算符使用了几次
        Array(){memset(items,0,sizeof(items));} // 使用memset函数初始化数组,需要知道首地址、初始化值、数组元素类型所占空间
        T& operator[] (int ii) // 重载[]运算符,返回值类型为T的引用
        {
            this->sign++; // this指针类似于python中的self,他默认标记类地址,可以查找类内容
            return *(items+ii);
        }
        const T& operator[] (int ii) // 如果是const修饰的数组,上面的函数将不会被调用
        const{
            return *(items+ii);
        }
};

再来测试一下:

int main()
{
    // Array<double,5> a;
    Array<double> a;
    a[0]=1;a[1]=2;a[2]=3;a[3]=4;
    for(int i=0;i<10;i++)
    {
        cout<<a[i]<<" ";
    }
    cout<<"\n"<<a.sign<<endl;
}
// 输出为:1 2 3 4 0 0 0 0 0 0 
//		  14

注意:非通用类型参数通常是整数型(但不绝对),这个值在函数模板实例化之后是不可以修改的。定长数组功能实现起来并不难,接下来我们试试变长数组。

可变数组

这个功能并不难实现,我们仅对上述代码实现少量改动,即当访问下标超过数组最大长度时,我们要重新分配更大的内存存储这个数组:

template <class T>
class Vector
{
    private:
        int len;
        T *items; // 声明一个指针变量,用于接收动态数组
    public:
        Vector(int size=5) // C++的参数也可以像python一样使用默认参数
        {
            len=size;
            items=new T[len];
        } 
        ~Vector() // 释放动态分配的内存
        {
            delete[] items;
            items=nullptr;
        }
        void resize(int size) // 当数组超出容量时会重新分配内存
        {
            T* temp;
            if (size>len)
            {temp=new T[size];} // 分配更大的内存
            else return;
            for(int i=0;i<len;i++) // 将旧数组拷贝到新数组
            {temp[i]=items[i];}
            delete[] items;        // 释放原数组
            items=temp;            // 指向新数组
            len=size;              // 更新数组大小
        }
        int size()const{ // 返回数组长度
            return len;
        }
        T& operator[] (int ii) // 重载[]运算符,返回值类型为T的引用
        {
            if (ii>=len){
                this->resize(len+5); // 分配多五个空间的新数组
            }
            return *(items+ii);
        }
        const T& operator[] (int ii) // 如果是const修饰的数组,上面的函数将不会被调用
        const{
            return *(items+ii);
        }
};

这样就完成了,大家可以自行尝试一下。

2.使用类模板实现栈结构

栈是一种先进后出的数据结构,我们主要通过类模板实现它的入栈、出栈、判空、判满功能。首先我们编写一个函数实现int类型的栈:

class Stack
{
    private:
        int *items; // 由数组结构定义的栈
        int stacksize; // 栈含元素数量
        int top; // 栈顶指针,并非指针类型,而是类似于数组下标
    public:
        // 初始化栈
        Stack(int size):stacksize(size),top(0)
        {
            items=new int[stacksize];
        }
        // 析构函数,删除动态空间
        ~Stack()
        {
            delete[] items;
            items=nullptr;
        }
        // 判空函数,即判断栈顶指针是否为0
        bool isempty() const // 不允许在函数内修改任何变量值
        {
            return top==0;
        }
        // 判满函数
        bool isfull() const
        {
            return top==stacksize;
        }
        // 压栈函数
        bool push(const int& item)
        {
            if(!isfull())
            {
                items[top++]=item;
                return true;
            }
            else return false;
        }
        // 出栈函数
        bool pop(int &item)
        {
            if(!isempty())
            {
                item=items[--top];
                return true;
            }
            else return false;
        }
};

这样我们就实现了定义一个功能简单的处理类型int栈。下面我们先来测试一下各个功能是否正常,然后再看看如何把这个具体类制作成类模板:

int main()
{
    Stack ss(5);
    cout<<ss.isempty()<<" "<<ss.isfull()<<endl;
    ss.push(1);ss.push(2);ss.push(3);ss.push(4);ss.push(5);
    cout<<ss.isempty()<<" "<<ss.isfull()<<endl;
    int item;
    while(ss.pop(item))
    {
        cout<<item<<" ";
    }
}
// 输出为:1 0
//		  0 1
//		  5 4 3 2 1

这样看起来,我们的功能实现的还不错。接下来我们把它定义成可以处理不同数据类型的栈类。我们可以使用

typedef *** Mydata

在星号处填入想要处理的类型,然后将与items有关的代码全都改成Mydata(或Mydata*)类。这样做的缺点是我们经常要手动去调整类所处理的数据类型。
在有了类模板之后,我们就可以更方便的解决这些问题了:

template<class T>
class Stack
{
    private:
        T *items; // 由数组结构定义的栈
        int stacksize; // 栈含元素数量
        int top; // 栈顶指针,并非指针类型,而是类似于数组下标
    public:
        // 初始化栈
        Stack(int size):stacksize(size),top(0)
        {
            items=new T[stacksize];
        }

		// 判空和判满不需要改动

		 // 压栈函数
        bool push(const T& item)
        {
            if(!isfull())
            {
                items[top++]=item;
                return true;
            }
            else return false;
        }
        // 出栈函数
        bool pop(T &item)
        {
            if(!isempty())
            {
                item=items[--top];
                return true;
            }
            else return false;
        }

这样我们就成功的做出了一个模板类实现栈的部分功能了。


http://www.kler.cn/a/228670.html

相关文章:

  • 论文阅读笔记:AI+RPA
  • Spring6.0新特性-HTTP接口:使用@HttpExchange实现更优雅的Http客户端
  • 创建一个简单的spring boot+vue前后端分离项目
  • OpenWrt 中使用 LuCI 界面部署 Docker 镜像
  • 将IDLE里面python环境pyqt5配置的vscode
  • SQL-leetcode—626. 换座位
  • Python调用pyspark报错整理
  • Class 类
  • SpringBoot实战项目第一天
  • 为什么选择AGPL3.0开源协议
  • ROS从入门到精通4-1:Docker安装与常用命令总结
  • Windows自动化实现:系统通知和任务栏图标自定义
  • jmeter-04创建请求
  • 类银河恶魔城学习记录1-5 CollisionCheck源代码 P32
  • 2024 高级前端面试题之 性能优化模块 「精选篇」
  • 华为机考入门python3--(8)牛客8-合并表记录
  • RedHat8.4安装邮件服务器
  • Redis核心技术与实战【学习笔记】 - 17.Redis 缓存异常:缓存雪崩、击穿、穿透
  • BUG:docker启动之后直接退出问题
  • C++面试:数据库的连接池管理
  • docker-compose部署laravel项目实战(主机nginx连接项目容器)(详细配置过程)
  • SpringBoot 集成 WebSocket,实现后台向前端推送信息
  • 利用jmeter完成简单的压力测试
  • 贪心算法(简单易懂,考研复试上机知识点)
  • 保护个人信息安全,避免成为“互联网中的裸泳者”
  • 代码随想录算法训练营第27天| 39. 组合总和、40.组合总和II、131.分割回文串