【C++】vector 类模拟实现:探索动态数组的奥秘

news2024/11/18 6:57:03

🌟 快来参与讨论💬,点赞👍、收藏⭐、分享📤,共创活力社区。🌟

如果你对string,vector还存在疑惑,欢迎阅读我之前的作品 : 

之前文章🔥🔥🔥 【C++】vector 类深度解析:探索动态数组的奥秘

               🔥🔥🔥 【C++】string 类深度解析:探秘字符串操作的核心


目录

一、引言🎉

二、vector 类的功能需求分析👀

(一)存储元素数据📦

(二)元素访问与修改✍️

(三)元素数量相关操作📏

(四)迭代器支持🔍

(五)内存管理🧹

三、模拟实现的关键步骤和代码解析💻

(一)类的定义🎯

(二)构造函数实现🔨

(三)析构函数实现🚮

(四)获取元素数量和容量函数📏

(五)判断是否为空函数🤔

(六)预留空间函数📦

(七)调整大小函数📏

(八)下标操作符重载🎯

(九)赋值操作符重载✍️

(十)添加和删除元素函数🎯

(十一)迭代器相关函数🔍

四、总结😎


一、引言🎉

在 C++ 的编程世界里,动态数组就像一个神奇的魔法盒子🎁,可以根据我们的需求灵活地改变大小。而标准库中的 vector 类,就是这个魔法盒子的绝佳实现者🧙‍♂️!它为我们提供了超级方便的操作接口,让我们能够轻松地处理各种动态数据存储和操作的任务。你是不是也像我一样,对 vector 类的内部实现充满了好奇呢😏?

嘿嘿,那就跟着我一起踏上模拟实现 vector 类的奇妙之旅吧🚀! 


二、vector 类的功能需求分析👀

(一)存储元素数据📦

我们希望 vector 类能够像一个智能的容器一样🧺,用连续的内存空间来存放元素。这样的话,就可以像在一排整齐排列的小格子里找东西一样,通过索引快速地访问到任何一个元素😃,是不是很方便呢🧐?而且哦,它还要能自动调整存储空间的大小,就像一个会伸缩的魔法口袋👝,当我们添加或删除元素的时候,它能够自动适应,不会让内存出现浪费或者不够用的情况,是不是超级厉害😎!

(二)元素访问与修改✍️

  1. 当然啦,我们需要能够轻松地通过索引来获取元素的值,就像打开小格子拿东西一样简单😏。而且,我们还希望能够修改这个元素的值,让我们可以随心所欲地更新数据😎。
  2. 有时候,我们想要在指定的位置插入一个新元素,这就好比在排队的人群中突然插入一个小伙伴👫,vector 类要能够巧妙地把后面的元素都往后挪一挪,给新元素腾出空间哦😃。
  3. 还有还有,当我们想要删除某个位置的元素时,vector 类要能像变魔术一样,把后面的元素都往前移一移,填补上这个空缺,保持队伍的整齐有序😎。

(三)元素数量相关操作📏

  1. 我们得知道 vector 里面到底装了多少个元素呀,这样才能更好地控制和处理数据😉。所以,需要有一个方法来获取元素的数量。
  2. 当我们预先知道大概要存储多少个元素的时候,希望能够提前告诉 vector 类,让它提前准备好足够的空间,就像告诉魔法口袋我们大概要装多少东西一样,这样可以提高效率哦😎。

(四)迭代器支持🔍

为了能够方便地遍历 vector 中的元素,我们需要 vector 类提供迭代器。迭代器就像是一个小导游👨‍✈️,可以带着我们逐个访问 vector 中的元素,让遍历操作变得轻松愉快😃。

(五)内存管理🧹

最后但同样重要的是,vector 类要能够合理地管理内存。它要知道什么时候该申请新的内存,什么时候该释放不再使用的内存,就像一个勤劳的小管家一样,把内存管理得井井有条,避免内存泄漏和浪费,确保我们的程序能够高效稳定地运行😎。


三、模拟实现的关键步骤和代码解析💻

(一)类的定义🎯

template<typename T>
class MyVector {
private:
    T* data;           // 存储元素的数组指针
    size_t size_;      // 当前元素数量
    size_t capacity_;  // 数组容量

public:
    MyVector();                            // 默认构造函数
    MyVector(size_t n, const T& value = T());  // 构造函数,初始化n个相同元素
    MyVector(const MyVector<T>& other);     // 拷贝构造函数
    ~MyVector();                           // 析构函数

    size_t size() const;                    // 获取元素数量
    size_t capacity() const;                // 获取数组容量
    bool empty() const;                     // 判断是否为空
    void reserve(size_t new_capacity);      // 预留空间
    void resize(size_t new_size, T value = T());  // 调整大小

    T& operator[](size_t index);            // 下标操作符重载(非const版本)
    const T& operator[](size_t index) const;  // 下标操作符重载(const版本)
    MyVector<T>& operator=(const MyVector<T>& other);  // 赋值操作符重载

    void push_back(const T& value);         // 在末尾添加元素
    void pop_back();                        // 删除末尾元素
    void insert(size_t pos, const T& value);  // 在指定位置插入元素
    void erase(size_t pos);                 // 删除指定位置元素

    typedef T* iterator;                    // 迭代器类型定义
    iterator begin();                       // 返回起始迭代器
    iterator end();                         // 返回结束迭代器
};

 👇解释:

这里我们定义了一个模板类MyVector,它可以存储任意类型的元素哦😎。有一个指针data用来指向存储元素的数组,size_记录当前元素的数量,capacity_表示数组的容量。然后还有各种各样的函数,它们就像一群小助手,分别负责不同的操作任务呢😉。

 

(二)构造函数实现🔨

  1. 默认构造函数
template<typename T>
MyVector<T>::MyVector() : data(nullptr), size_(0), capacity_(0) {}

😃这个默认构造函数超级简单,就像创建一个空的魔法口袋一样,把data指针设为nullptr,表示还没有分配内存,元素数量size_和容量capacity_都初始化为 0😎。
2. 指定大小和初始值的构造函数

template<typename T>
MyVector<T>::MyVector(size_t n, const T& value) : size_(n), capacity_(n) {
    data = new T[capacity_];
    for (size_t i = 0; i < size_; ++i) {
        data[i] = value;
    }
}

👀这个构造函数就像是按照我们的要求定制魔法口袋的大小和初始物品一样🧐。我们告诉它要创建一个能装n个元素的 vector,并且每个元素都初始化为value。它先分配足够的内存空间,然后用一个循环把每个元素都设置为指定的值😉。
3. 拷贝构造函数

template<typename T>
MyVector<T>::MyVector(const MyVector<T>& other) : size_(other.size_), capacity_(other.capacity_) {
    data = new T[capacity_];
    for (size_t i = 0; i < size_; ++i) {
        data[i] = other.data[i];
    }
}

😎拷贝构造函数就像是复制一个一模一样的魔法口袋哦😏。它创建一个新的 vector,大小和容量都和传入的other一样,然后把other中的元素一个一个地复制到新的 vector 中,这样新的 vector 就和原来的一模一样啦😉。

(三)析构函数实现🚮

template<typename T>
MyVector<T>::~MyVector() {
    if (data!= nullptr) {
        delete[] data;
    }
}

👀这个析构函数就像一个勤劳的清洁工🧹,当 vector 对象生命周期结束的时候,它会检查data指针是否为空。如果不为空,就释放掉data所指向的内存,避免内存泄漏,确保我们的程序干干净净、健健康康😎。

(四)获取元素数量和容量函数📏

1.获取元素数量函数

template<typename T>
size_t MyVector<T>::size() const {
    return size_;
}

😃这个函数就像一个小计数器一样,直接返回当前 vector 中元素的数量size_,简单又直接😎。
2. 获取容量函数

template<typename T>
size_t MyVector<T>::capacity() const {
    return capacity_;
}

👀它也是一样的简单,直接把数组的容量capacity_返回给我们,让我们知道这个魔法口袋最多能装多少东西😉。

(五)判断是否为空函数🤔

template<typename T>
bool MyVector<T>::empty() const {
    return size_ == 0;
}

😎这个函数就像一个小侦探🕵️‍♂️,检查元素数量size_是否为 0。如果是 0,那就说明 vector 是空的,就像魔法口袋里什么都没有一样,返回true;否则返回false😉。

(六)预留空间函数📦

template<typename T>
void MyVector<T>::reserve(size_t new_capacity) {
    if (new_capacity > capacity_) {
        T* new_data = new T[new_capacity];
        for (size_t i = 0; i < size_; ++i) {
            new_data[i] = data[i];
        }
        if (data!= nullptr) {
            delete[] data;
        }
        data = new_data;
        capacity_ = new_capacity;
    }
}

👀这个函数就像是给魔法口袋升级扩容一样😃!当我们预计要装更多元素的时候,就可以调用它。如果新的容量new_capacity比当前容量大,它就会分配一块新的更大的内存空间new_data,然后把原来的元素都复制到新空间里。最后,释放原来的内存,把data指针指向新的内存空间,并且更新容量capacity_,这样 vector 就有了更大的空间来装元素啦😎。

(七)调整大小函数📏

template<typename T>
void MyVector<T>::resize(size_t new_size, T value) {
    if (new_size > size_) {
        if (new_size > capacity_) {
            reserve(new_size);
        }
        for (size_t i = size_; i < new_size; ++i) {
            data[i] = value;
        }
    } else if (new_size < size_) {
        for (size_t i = new_size; i < size_; ++i) {
            data[i].~T();  // 手动调用析构函数
        }
    }
    size_ = new_size;
}

😉这个函数就像是重新调整魔法口袋的大小和里面的物品数量一样🧐。如果新的大小new_size比当前元素数量size_大,它会先检查容量是否足够,如果不够就调用reserve函数扩容。然后,用指定的值value填充新增加的位置。如果新的大小比当前数量小,它会手动调用析构函数来清理多余的元素,最后更新元素数量size_😎。

 

(八)下标操作符重载🎯

  1. 非 const 版本
template<typename T>
T& MyVector<T>::operator[](size_t index) {
    return data[index];
}

😃这个非 const 版本的下标操作符重载就像一把神奇的钥匙🔑,可以让我们通过索引直接访问和修改 vector 中的元素。它返回data[index]的引用,这样我们就可以像操作普通变量一样对元素进行读写操作啦😎。
2. const 版本

template<typename T>
const T& MyVector<T>::operator[](size_t index) const {
    return data[index];
}

👀const 版本的下标操作符重载就像是一把只能看不能改的钥匙🔐,当我们有一个 const 的 vector 对象时,通过这个操作符只能读取元素的值,不能修改,确保了 const 对象的安全性😉。

 

(九)赋值操作符重载✍️

template<typename T>
MyVector<T>& MyVector<T>::operator=(const MyVector<T>& other) {
    if (this!= &other) {
        if (data!= nullptr) {
            delete[] data;
        }
        size_ = other.size_;
        capacity_ = other.capacity_;
        data = new T[capacity_];
        for (size_t i = 0; i < size_; ++i) {
            data[i] = other.data[i];
        }
    }
    return *this;
}

😎这个赋值操作符重载就像是把一个魔法口袋里的东西全部复制到另一个魔法口袋里一样🧙‍♂️。首先,它会检查是不是自我赋值,如果不是,就先释放掉原来的内存。然后,复制other的大小、容量和元素到当前 vector 中,最后返回当前对象的引用,这样就可以连续赋值啦😉。

(十)添加和删除元素函数🎯

  1. 在末尾添加元素函数
template<typename T>
void MyVector<T>::push_back(const T& value) {
    if (size_ == capacity_) {
        reserve(capacity_ == 0? 1 : capacity_ * 2);
    }
    data[size_++] = value;
}

😃这个函数就像是往魔法口袋的末尾放一个新东西一样🎁。它先检查口袋是否还有空间,如果没有了,就调用reserve函数扩容(如果当前容量是 0,就扩为 1,否则扩为原来的两倍)。然后,把新元素放到data[size_]的位置,并且把元素数量size_加 1😎。
2. 删除末尾元素函数

template<typename T>
void MyVector<T>::pop_back() {
    if (size_ > 0) {
        --size_;
        data[size_].~T();  // 手动调用析构函数
    }
}

👀这个函数就像是从魔法口袋的末尾拿出一个东西一样😉。如果 vector 不为空,它就把元素数量size_减 1,并且手动调用析构函数来清理最后一个元素,就像把拿出来的东西处理掉一样😎。
3. 在指定位置插入元素函数

template<typename T>
void MyVector<T>::insert(size_t pos, const T& value) {
    if (pos > size_) {
        return;
    }
    if (size_ == capacity_) {
        reserve(capacity_ == 0? 1 : capacity_ * 2);
    }
    for (size_t i = size_; i > pos; --i) {
        data[i] = data[i - 1];
    }
    data[pos] = value;
    ++size_;
}

😎这个函数就像是在排队的人群中插入一个小伙伴一样👫。它先检查插入位置pos是否合法,如果合法,就检查容量是否足够,不够就扩容。然后,把pos及后面的元素都往后移一位,腾出空间,把新元素插入到pos位置,最后把元素数量size_加 1😉。
4. 删除指定位置元素函数

template<typename T>
void MyVector<T>::erase(size_t pos) {
    if (pos >= size_) {
        return;
    }
    for (size_t i = pos; i < size_ - 1; ++i) {
        data[i] = data[i + 1];
    }
    --size_;
    data[size_].~T();  // 手动调用析构函数
}

👀这个函数就像是从排队的人群中请走一个小伙伴一样😉。它先检查删除位置pos是否合法,如果合法,就把pos后面的元素都往前移一位,覆盖掉要删除的元素,然后把元素数量size_减 1,并且手动调用析构函数来清理最后一个元素😎。

(十一)迭代器相关函数🔍

1.迭代器类型定义

template<typename T>
typedef T* MyVector<T>::iterator;

😃这里我们定义了迭代器的类型,其实就是一个指向T类型的指针,这样我们就可以用这个指针来遍历 vector 中的元素啦😉。
2. 起始迭代器函数

template<typename T>
typename MyVector<T>::iterator MyVector<T>::begin() {
    return data;
}

👀这个函数就像是给我们一个指向 vector 开头的小箭头一样,它返回data指针,也就是指向第一个元素的位置,让我们可以从这里开始遍历元素😎。
3. 结束迭代器函数

template<typename T>
typename MyVector<T>::iterator MyVector<T>::end() {
    return data + size_;
}

😉这个函数就像是给我们一个指向 vector 末尾后面一个位置的小箭头,它返回data + size_,表示遍历到这个位置就结束了,因为这个位置并不存储实际元素,只是一个结束的标志😎。


四、总结😎

通过模拟实现 vector 类,我们仿佛走进了动态数组的魔法世界🧙‍♂️,深入了解了其背后的实现原理和技术细节。从存储结构的精心设计,到各种操作函数的巧妙实现,再到内存管理的严谨把控,每一个环节都像是魔法世界里的一块拼图🧩,缺一不可。

希望大家继续保持探索的热情,不断挖掘编程世界里的更多奥秘😉!加油哦💪!


以后我将深入研究继承、多态、模板等特性,并将默认成员函数与这些特性结合,以解决更复杂编程问题!欢迎关注我👉【A Charmer】 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2242662.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

小程序-基于java+SpringBoot+Vue的驾校预约平台设计与实现

项目运行 1.运行环境&#xff1a;最好是java jdk 1.8&#xff0c;我们在这个平台上运行的。其他版本理论上也可以。 2.IDE环境&#xff1a;IDEA&#xff0c;Eclipse,Myeclipse都可以。推荐IDEA; 3.tomcat环境&#xff1a;Tomcat 7.x,8.x,9.x版本均可 4.硬件环境&#xff1a…

初识算法 · 模拟(1)

目录 前言&#xff1a; 替换所有的问号 题目解析 算法原理 算法编写 提莫攻击 题目解析 算法原理 算法编写 外观数列 题目解析 算法原理 算法编写 前言&#xff1a; ​本文的主题是模拟&#xff0c;通过三道题目讲解&#xff0c;一道是提莫攻击&#xff0c;一道是…

使用 Vue 和 Create-Vue 构建工程化前端项目

目录 前言1. 工程化的意义与 Vue 的生态支持2. 搭建 Vue 工程化项目2.1 环境准备2.2 使用 create-vue 创建项目2.2.1 初始化项目2.2.2 安装依赖2.2.3 本地运行 3. Vue 项目的目录结构解析4. Vue 开发流程详解4.1 项目入口与根组件4.1.1 main.js 的作用4.1.2 App.vue 的结构 4.2…

Android中的AMS(Activity Manager Service)详解

Android中的AMS&#xff08;Activity Manager Service&#xff09;详解 AMS (Activity Manager Service) 是 Android 系统中非常核心的服务之一&#xff0c;它负责管理应用程序的生命周期、任务栈、进程、广播、服务等功能。AMS 是整个 Android Framework 的调度中心&#xff…

31.3 XOR压缩和相关的prometheus源码解读

本节重点介绍 : xor 压缩value原理xor压缩过程讲解xor压缩prometheus源码解读xor 压缩效果 xor 压缩value原理 原理:时序数据库相邻点变化不大&#xff0c;采用异或压缩float64的前缀和后缀0个数 xor压缩过程讲解 第一个值使用原始点存储计算和前面的值的xor 如果XOR值为0&…

UNIAPP发布小程序调用讯飞在线语音合成+实时播报

语音合成能够将文字转化为自然流畅的人声&#xff0c;提供100发音人供您选择&#xff0c;支持多语种、多方言和中英混合&#xff0c;可灵活配置音频参数。广泛应用于新闻阅读、出行导航、智能硬件和通知播报等场景。 在当下大模型火爆的今日&#xff0c;语音交互页离不开语音合…

【蓝牙协议栈】【BLE】【BAS】精讲蓝牙电池服务

1. 蓝牙电池服务(Bluetooth Battery Service)概念 蓝牙电池服务是蓝牙设备与其他设备通信时用于报告其剩余电池电量的标准服务。它让用户能够随时了解蓝牙设备(如无线耳机、智能手表、蓝牙鼠标/键盘等)的电池状态,从而方便地管理这些设备的续航与电源使用。 BAS通常用于在…

无线迷踪:陈欣的网络之旅

第一章 陈欣是一名资深的网络工程师&#xff0c;工作在一家领先的科技公司。她的生活平静而有序&#xff0c;直到有一天&#xff0c;公司的无线网络突然出现了严重的问题。员工们的设备频繁断开连接&#xff0c;无法正常使用。这个问题不仅影响了工作效率&#xff0c;还引起了…

【redis】—— 环境搭建教程

上一节&#xff0c;我们大致了解了Redis的几个重要版本&#xff0c;在本教程中&#xff0c;我们选择了5.0版本&#xff0c;因为5.0已经具备了大部分的功能特性&#xff0c;并且与7.0版本相比&#xff0c;其安装使用过程更为简便。 Redis的官方并不直接支持微软的Windows操作系统…

如何查看python源代码

众所周知&#xff0c;Python内建了许多函数模块&#xff0c;并且我们可能还会安装许多第三方模块等等。 下面以getpass为例查看其源代码。 1.help(getpass) 输入该命令找到file路径&#xff0c;并且可以查看其其提供的功能。 2.利用getpass.__file__查看位置 最后找到该文件…

java笔试练习题笔记(10)

关于继承和实现说法正确的 是 &#xff1f; ( )A.类可以实现多个接口&#xff0c;接口可以继承&#xff08;或扩展&#xff09;多个接口 B.类可以实现多个接口&#xff0c;接口不能继承&#xff08;或扩展&#xff09;多个接口 C.类和接口都可以实现多个接口 D.类和接口都不…

前端开发之打印功的使用和实例(vue-print-nb)

通过插件来进行实现 前言效果图1、安装插件vue2vue32、 引入Vue项目2、 使用2.1、在项目中创建按钮并且使用v-print绑定绑定打印事件2.2、编写要打印的内容,给内容附加唯一的id2.3、绑定的时间的方法和参数3、整体代码(此代码是通过vue3来进行实现的但是逻辑都是一样的)前言…

NavVis VLX3的精度怎么去进行验证?【上海沪敖3D】

01、精度评价现状 三维捕捉行业还没有建立一个用于估算或验证移动激光扫描系统精度的统一标准。因此&#xff0c;需要高精度交付成果的专业人士很难相信设备所标注的精度规格&#xff0c;也就很难知道基于SLAM的移动激光扫描系统是否适合当前的项目。 NavVis将通过展示一种严格…

Java | Leetcode Java题解之第564题寻找最近的回文数

题目&#xff1a; 题解&#xff1a; class Solution {public String nearestPalindromic(String n) {long selfNumber Long.parseLong(n), ans -1;List<Long> candidates getCandidates(n);for (long candidate : candidates) {if (candidate ! selfNumber) {if (ans…

ES6标准-Promise对象

目录 Promise对象的含义 Promise对象的特点 Promise对象的缺点 Promise对象的基本用法 Promise对象的简单例子 Promise新建后就会立即执行 Promise对象回调函数的参数 Promise参数不会中断运行 Promise对象的then方法 Promise对象的catch()方法 Promise状态为resolv…

如何利用CSS制作导航菜单

1.利用CSS技术&#xff0c;结合链接和列表&#xff0c;设计并实现“山水之间”页面 示例代码 <!DOCTYPE html> <html><head><meta charset"utf-8"><title>山水之间</title><style>.all{width:900px;}.top{width:900px;h…

Github 2024-11-17 php开源项目日报 Top10

根据Github Trendings的统计,今日(2024-11-17统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量PHP项目10JavaScript项目2Nextcloud服务器:安全的数据之家 创建周期:2796 天开发语言:PHP, JavaScript协议类型:GNU Affero General Public…

【工具变量】2024-2025年地级市异质性数据、城市分组异质性数据(老工业、环境保护、人口流入、沿海等)

一、数据范围&#xff1a; &#xff08;1&#xff09;南北方城市 &#xff08;2&#xff09;东中西城市 &#xff08;3&#xff09;七大地理区、八大综合经济区 &#xff08;4&#xff09;城市群&#xff0c;长三角珠三角京津冀等 &#xff08;5&#xff09;长江流域沿岸、黄河…

响应式网页设计--css

CSS&#xff08;Cascading Style Sheets&#xff0c;层叠样式表&#xff09;是用来控制网页的外观和布局的语言。它与 HTML 一起工作&#xff0c;通过样式定义网页元素的显示方式。CSS 可以控制多种视觉效果&#xff0c;如字体、颜色、布局、间距等。 基本语法&#xff1a; C…

定时器的小应用

第一个项目 第一步&#xff0c;RCC开启时钟&#xff0c;这个基本上每个代码都是第一步&#xff0c;不用多想&#xff0c;在这里打开时钟后&#xff0c;定时器的基准时钟和整个外设的工作时钟就都会同时打开了 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);第二步&…