vector【2】模拟实现(超详解哦)

vector

  • 引言(实现概述)
  • 接口实现详解
    • 默认成员函数
      • 构造函数
      • 析构函数
      • 赋值重载
    • 迭代器
    • 容量
      • size与capacity
      • reserve
      • resize
      • empty
    • 元素访问
    • 数据修改
      • insert
      • erase
      • push_back与pop_back
      • swap
  • 模拟实现源码概览
  • 总结

引言(实现概述)

在前面,我们介绍了vector的使用:
戳我康vector介绍与使用

在本篇文章中将重点介绍vector的接口实现,通过模拟实现可以更深入的理解与使用vector。
在这里插入图片描述

我们可以在网上搜索到vector的实现源码, 与string中使用一个指针指向存储数据的空间,两个整型来刻画size与capacity不同,vector中是通过三个迭代器 _start_finish_endOfStorage分别指向数据块的起始位置、有效数据末尾的下一个位置、存储容量末尾的下一个位置来管理数据的。vector中迭代器就是原生指针,本质上就是使用三个指针来管理动态申请的存储数据的空间。

在这里插入图片描述

vector是一个类模板,其声明与定义不能分离。我们将模拟实现的vector放在我们创建的命名空间内,以防止与库发生命名冲突。

在vector的模拟实现中,我们只实现一些主要的接口,包括默认成员函数、迭代器、容量、元素访问与数据修改

接口实现详解

默认成员函数

构造函数

构造函数的模拟实现包括无参构造、n个指定元素构造、迭代器区间构造与拷贝构造

无参构造:即首先在初始化列表中,将三个属性全部初始化为空指针即可:

    vector()
        : _start(nullptr)
        , _finish(nullptr)
        , _endOfStorage(nullptr)
    {}

n个指定元素构造

这个重载版本有两个参数,第一个是int,第二个是const T&,表示用nvalue构造vector,第二个参数缺省值为其默认构造T()

首先new一块大小为n个元素大小的空间,将其赋值给_start
然后_finish的值就是_start + n_endOfStorage的值与_finish相同;
最后for循环将nvalue写入空间中:

    vector(int n, const T& value = T())   //
    {
        _start = new T[n];
        _finish = _start + n;
        _endOfStorage = _finish;

        for (int i = 0; i < n; ++i)
        {
            *(_start + i) = value;
        }
    }

迭代器区间构造

使用迭代器区间的构造,是一个函数模板,即可以使用其他容器的迭代器区间来构造vector。

这个重载版本的实现有许多方式,这里的实现是偷懒版本的,即首先将三个属性初始化为空指针后,再复用push_back(后面实现)来将迭代器区间中的元素尾插到新vector中:

    template<class InputIterator>
    vector(InputIterator first, InputIterator last)
        : _start(nullptr)
        , _finish(nullptr)
        , _endOfStorage(nullptr)
    {
        while (first < last)
        {
            push_back(*first);
            ++first;
        }
    }

拷贝构造

拷贝构造时,首先将三个属性都初始化为空指针;
然后使用reserve(后面会实现)将新vector扩容与原vector一致;
最后循环将原vector中的数据拷贝到新vector中即可:

    vector(const vector<T>& v)
        : _start(nullptr)
        , _finish(nullptr)
        , _endOfStorage(nullptr)
    {
        int sz = v.size();
        reserve(sz);
        for (int i = 0; i < sz; ++i)
        {
            *(_start + i) = v[i];
        }
    }

析构函数

析构函数即释放动态申请的资源,即delete[] _start即可,同时可以顺便将三个属性均置空:

    ~vector()
    {
        if (_start != nullptr)
        {
            delete[] _start;
            _start = _finish = _endOfStorage = nullptr;
        }
    }

赋值重载

在实现赋值运算符重载时,存在深浅拷贝的问题,为了简便我们使用现代版本:

现代版本的参数类型为vector<T>,而不是引用,这就使得vector对象在传参时会生成一个临时对象,我们将这个临时对象与要替换的对象*this互换,就实现了将一个对象赋值到了*this,最后返回*this即可,临时对象会在函数栈帧销毁时析构(swap后面实现)。

    vector<T>& operator= (vector<T> v)
    {
        swap(v);
        return *this;
    }

迭代器

vector的迭代器本质上就是原生指针,所以我们只需要T* 重命名为iterator即可实现迭代器,并且具有原生指针的++--+-指针相减等的属性:

	typedef T* iterator;
    typedef const T* const_iterator;

与string部分相同,我们暂时只实现beginend,关于反向迭代器的实现在后面会详细介绍。
begin返回首元素的地址,end返回尾元素下一个位置的地址,他们分别重载有const版本:

    iterator begin()
    {
        return _start;
    }
    iterator end()
    {
        return _finish;
    }
    const_iterator begin() const
    {
        return _start;
    }
    const_iterator end() const
    {
        return _finish;
    }

容量

size与capacity

之前讲到vector迭代器的底层是原生指针,支持指针减指针的操作
对于size即元素的个数,_finish - _start的值即空间中数据的末尾的下一个位置的指针减首元素位置的指针,即元素个数;
对于capacity即容量的大小,_endOfStorage - _start的值即空间末尾下一个位置的指针减首元素位置的指针,即容量大小:

	size_t size() const
    {
        return _finish - _start;
    }
    size_t capacity() const
    {
        return _endOfStorage - _start;
    }

reserve

reserve用于扩容
对于C++而言,使用new扩容时,必须进行重新开辟空间,将原空间中的元素转移至新空间,最后释放原空间的操作。这样的过程将是十分影响效率的:

在库实现中,当传参的n大于原容量时,reserve会实现扩容,小于原容量时,reserve不进行缩容操作。所以我们模拟实现时,先判断n是否大于原容量,当大于原容量时在再进行后续操作;
首先new一块大小为n个元素大小的空间;
然后就需要挪动数据,需要注意的是,不能使用memcpy来拷贝数据到新空间中,因为memcpy是逐字节拷贝,而自定义类型是会有动态申请的资源的,这样在释放原空间时就会使新空间中的属性为野指针,当生命周期结束时释放资源时就会释放野指针从而崩溃。所以我们需要调用operator=逐一拷贝数据到新空间中,并释放原空间
最后令_start指向新空间的首元素,_finish指向数据的结尾(可以在释放前记录size的值,此时加上即可),_endOfStorage指向空间的结尾,即_start + n

 void reserve(size_t n)
    {
        if (n > capacity())
        {
            T* temp = new T[n];
            int sz = size();

            if (sz != 0)
            { 
                //当T为自定义类型时,memcpy为浅拷贝,temp中的自定义类型的数据与*this是相同的,delete调析构释放原空间就会使temp中数据为野指针
                //memcpy(temp, cbegin(), sizeof(T) * size()); 
                for (size_t i = 0; i < size(); ++i)
                {
                    *(temp + i) = *(_start + i);
                }

                delete[] _start;
            }
            _start = temp;
            _finish = _start + sz;
            _endOfStorage = _start + n;
        }
    }

resize

resize用于改变元素个数
n小于元素个数时,删除多于的元素;n大于元素个数时,使用指定的元素value补足(value为缺省参数,缺省值为T()

首先判断n是否大于元素个数,当大于元素个数时还需要进一步判断n是否大于容量需要扩容;
之后逐一用value补足,这里同样需要是用operator=,来避免浅拷贝带来的问题,并调整_finish的指向;
当小于元素个数时,直接将_finish的值调整为_start + n即可:

    void resize(size_t n, const T& value = T())
    {
        if (n > size())
        {
            if (n > capacity())
            {
                reserve(n);
            }
            int oldSize = size();
            _finish = _start + n;

            for (size_t i = oldSize; i < n; ++i)
            {
                *(_start + i) = value;
            }
        }
        else
        {
            _finish = _start + n;
        }
    }

empty

empty用于判断vector是否为空,为空返回true,否则返回false。这里复用size即可,当size返回0时即为空:

    bool empty()
    {
        if (size() == 0)
        {
            return true;
        }
        return false;
    }

元素访问

元素访问即实现operator[],可以实现通过下标访问元素
有两个重载版本即普通对象与const对象。

首先判断pos是否越界,因为pos为无符号整型,所以只需要判断_start + pos 是否小于 _finish即可;
然后直接返回_start + pos的解引用即可:

    T& operator[](size_t pos)
    {
        assert(_start + pos < _finish);
        return *(_start + pos);
    }
    const T& operator[](size_t pos) const
    {
        assert(_start + pos < _finish);
        return *(_start + pos);
    }

数据修改

insert

insert用于在pos位置插入数据,模拟实现insert时,我们只实现在pos位置(迭代器)插入一个元素的情况:

首先判断pos是否越界,没有越界时还需要再判断是否需要扩容;
这里就存在一个问题,在上一篇文章提到了迭代器失效的问题:扩容后,指向原来空间的迭代器pos就会成为野指针而失效。为解决这个问题,我们可以事先计算pos对于_start的相对位置sz,从而在释放原空间后通过这个相对位置在新空间中重新找到pos,即_start + sz
然后就可以循环将pos位置及以后的元素逐一向后移动一个元素。这个过程是十分影响效率的;
最后将要插入的元素放在pos位置,并++_finish

  	iterator insert(iterator pos, const T& x) //pos传参,在reserve后会出现迭代器失效
    {
        assert(pos >= _start && pos <= _finish);
        int sz = pos - _start;  

        if (size() >= capacity())
        {
            reserve(capacity() == 0 ? 10 : capacity() * 2);
            pos = _start + sz;  //解决迭代器失效
        }
        vector<T>::iterator it = end() - 1;
        while (it >= pos)
        {
            *(it + 1) = *it;
            --it;
        }
        *pos = x;
        ++_finish;

        return _start;
    }

erase

erase用于删除一段数据,这里只模拟实现删除pos位置(迭代器)的一个元素:

首先判断pos是否越界,如果没有越界再判断容器是否为空,为空就直接返回_start
然后循环将pos后面的元素逐一向前移动一个元素(从后向前覆盖);
最后--_finish,并返回_start

    iterator erase(iterator pos)
    {
        assert(pos >= _start && pos < _finish);
        if (empty())
        {
            return _start;
        }
        vector<T>::iterator it = pos + 1;
        while (it < _endOfStorage)
        {
            *(it - 1) = *it;
            ++it;
        }
        --_finish;

        return _start;
    }

push_back与pop_back

由于在任意位置插入与删除十分影响效率,头插与头删更甚,所以库中只提供了尾插与尾删的接口,不用挪动数据使得其效率很高

模拟实现时其实只需要调用上面实现的inserterase即可:
push_back即在end()的位置插入一个元素x
pop_back即在end() - 1的位置删除一个元素:

    void push_back(const T& x)
    {
        insert(end(), x);
    }
    void pop_back()
    {
        erase(end() - 1);
    }

swap

swap函数用于交换两个对象的数据

使用算法库中的swap通过创建临时变量交换的话,就会发生多次深拷贝,十分影响效率。
对于vector对象的交换,只需要逐一交换他们的三个属性即可:

	void swap(vector<T>& v)
    {
        std::swap(_start, v._start);
        std::swap(_finish, v._finish);
        std::swap(_endOfStorage, v._endOfStorage);
    }

模拟实现源码概览

(关于反向迭代器的实现在后面会详细介绍,现在可以暂时忽略)

#include<iostream>
#include<cassert>
#include"my_reverse_iterator.h"

namespace qqq
{
    template<class T>
    class vector
    {
    public:
        / iterator ///
        //vector的迭代器是一个原生指针 
        typedef T* iterator;
        typedef const T* const_iterator;
        typedef ReverseIterator<iterator, T&, T*> reverse_iterator;
        typedef ReverseIterator<const_iterator, const T&, const T*> const_reverse_iterator;


        iterator begin()
        {
            return _start;
        }
        iterator end()
        {
            return _finish;
        }
        const_iterator cbegin() const
        {
            return _start;
        }
        const_iterator cend() const
        {
            return _finish;
        }
        reverse_iterator rbegin()
        {
            return reverse_iterator(end());
        }
        reverse_iterator rend()
        {
            return reverse_iterator(begin());
        }
        /// construct and destroy //
        vector()
            : _start(nullptr)
            , _finish(nullptr)
            , _endOfStorage(nullptr)
        {}
        vector(int n, const T& value = T())   //
        {
            _start = new T[n];
            _finish = _start + n;
            _endOfStorage = _finish;

            for (int i = 0; i < n; ++i)
            {
                *(_start + i) = value;
            }
        }
        template<class InputIterator>
        vector(InputIterator first, InputIterator last)
            : _start(nullptr)
            , _finish(nullptr)
            , _endOfStorage(nullptr)
        {
            while (first < last)
            {
                push_back(*first);
                ++first;
            }
        }
        //vector(const vector<T>& v)
        //{
        //    _start = new T[v.size()];
        //    _finish = _start + v.size();
        //    _endOfStorage = _start + v.capacity();
        // 
        //    memcpy(begin(), v.cbegin(), sizeof(T) * v.size());
        //}
        vector(const vector<T>& v)
            : _start(nullptr)
            , _finish(nullptr)
            , _endOfStorage(nullptr)
        {
            int sz = v.size();
            reserve(sz);
            for (int i = 0; i < sz; ++i)
            {
                *(_start + i) = v[i];
            }
        }
        vector<T>& operator= (vector<T> v)
        {
            swap(v);
            return *this;
        }
        ~vector()
        {
            if (_start != nullptr)
            {
                delete[] _start;
                _start = _finish = _endOfStorage = nullptr;
            }
        }

         capacity ///
        size_t size() const
        {
            return _finish - _start;
        }
        size_t capacity() const
        {
            return _endOfStorage - _start;
        }
        bool empty()
        {
            if (size() == 0)
            {
                return true;
            }
            return false;
        }
        void reserve(size_t n)
        {
            if (n > capacity())
            {
                T* temp = new T[n];
                int sz = size();

                if (sz != 0)
                { 
                    //当T为自定义类型时,memcpy为浅拷贝,temp中的自定义类型的数据与*this是相同的,delete调析构释放原空间就会使temp中数据为野指针
                    //memcpy(temp, cbegin(), sizeof(T) * size()); 
                    for (size_t i = 0; i < size(); ++i)
                    {
                        *(temp + i) = *(_start + i);
                    }

                    delete[] _start;
                }
                _start = temp;
                _finish = _start + sz;
                _endOfStorage = _start + n;
            }
        }
        void resize(size_t n, const T& value = T())
        {
            if (n > size())
            {
                if (n > capacity())
                {
                    reserve(n);
                }
                int oldSize = size();
                _finish = _start + n;

                for (size_t i = oldSize; i < n; ++i)
                {
                    *(_start + i) = value;
                }
            }
            else
            {
                _finish = _start + n;
            }
        }

        ///access

        T& operator[](size_t pos)
        {
            assert(_start + pos < _finish);
            return *(_start + pos);
        }
        const T& operator[](size_t pos) const
        {
            assert(_start + pos < _finish);
            return *(_start + pos);
        }


        ///modify/
        void push_back(const T& x)
        {
            insert(end(), x);
        }
        void pop_back()
        {
            erase(end() - 1);
        }
        void swap(vector<T>& v)
        {
            std::swap(_start, v._start);
            std::swap(_finish, v._finish);
            std::swap(_endOfStorage, v._endOfStorage);
        }
        iterator insert(iterator pos, const T& x)//pos传参,在reserve后会出现迭代器失效
        {
            assert(pos >= _start && pos <= _finish);
            int sz = pos - _start;  

            if (size() >= capacity())
            {
                reserve(capacity() == 0 ? 10 : capacity() * 2);
                pos = _start + sz;  //解决迭代器失效
            }
            vector<T>::iterator it = end() - 1;
            while (it >= pos)
            {
                *(it + 1) = *it;
                --it;
            }
            *pos = x;
            ++_finish;

            return _start;
        }
        iterator erase(iterator pos)
        {
            assert(pos >= _start && pos < _finish);
            if (empty())
            {
                return _start;
            }
            vector<T>::iterator it = pos + 1;
            while (it < _endOfStorage)
            {
                *(it - 1) = *it;
                ++it;
            }
            --_finish;

            return _start;
        }

    private:
        iterator _start;        // 指向数据块的开始
        iterator _finish;       // 指向有效数据的尾
        iterator _endOfStorage; // 指向存储容量的尾

    };
}

总结

到此,关于vector的主要接口实现就结束了
相信通过接口的模拟实现可以使我们更深入的了解vector
关于STL容器的介绍才刚刚开始,欢迎大家持续关注哦

如果大家认为我对某一部分没有介绍清楚或者某一部分出了问题,欢迎大家在评论区提出

如果本文对你有帮助,希望一键三连哦

希望与大家共同进步哦

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

企业计算机服务器遭到了locked勒索病毒攻击如何解决,勒索病毒解密

网络技术的不断发展&#xff0c;也为网络安全埋下了隐患&#xff0c;近期&#xff0c;我们收到很多企业的求助&#xff0c;企业的计算机服务器遭到了locked勒索病毒的攻击&#xff0c;导致企业的财务系统内的所有数据被加密无法读取&#xff0c;严重影响了企业的正常运行。最近…

设计模式十七:迭代器模式(Iterator Pattern)

迭代器模式&#xff08;Iterator Pattern&#xff09;是一种行为型设计模式&#xff0c;它提供了一种访问聚合对象&#xff08;例如列表、集合、数组等&#xff09;中各个元素的方法&#xff0c;而无需暴露其内部表示。迭代器模式将遍历元素和访问元素的责任分离开来&#xff0…

ArrayList

目录 1.ArrayList简介 2.ArrayList的构造 2.1ArrayList() 2.2ArrayList(Collection c) 2.3ArrayList(int initialCapacity) 3.ArrayList常见操作 4.ArrayList的遍历的遍历 1.ArrayList简介 在集合框架中&#xff0c; ArrayList 是一个普通的类&#xff0c;实现了 List…

【Java基础】Java对象的生命周期

【Java基础】Java对象的生命周期 一、概述 一个类通过编译器将一个Java文件编译为Class字节码文件&#xff0c;然后通过JVM中的解释器编译成不同操作系统的机器码。虽然操作系统不同&#xff0c;但是基于解释器的虚拟机是相同的。java类的生命周期就是指一个class文件加载到类…

学习笔记整理-DOM-02-事件监听

一、什么是"事件监听" DOM允许书写JavaScript代码以让HTML元素对事件作出反应什么是"事件": 用户与网页的交互动作当用户点击元素时当鼠标移动到元素上时当文本框的内容被改变时当键盘在文本框中被按下时当网页已加载完毕时… “监听”&#xff0c;顾名思义…

6.利用matlab完成 符号矩阵的秩和 符号方阵的逆矩阵和行列式 (matlab程序)

1.简述 利用M文件建立矩阵 对于比较大且比较复杂的矩阵&#xff0c;可以为它专门建立一个M文件。下面通过一个简单例子来说明如何利用M文件创建矩阵。 例2-2 利用M文件建立MYMAT矩阵。(1) 启动有关编辑程序或MATLAB文本编辑器&#xff0c;并输入待建矩阵&#xff1a;(2) 把…

Centos 防火墙命令

查看防火墙状态 systemctl status firewalld.service 或者 firewall-cmd --state 开启防火墙 单次开启防火墙 systemctl start firewalld.service 开机自启动防火墙 systemctl enable firewalld.service 重启防火墙 systemctl restart firewalld.service 防火墙设置开…

【mysql】事务的四种特性的理解

&#x1f307;个人主页&#xff1a;平凡的小苏 &#x1f4da;学习格言&#xff1a;命运给你一个低的起点&#xff0c;是想看你精彩的翻盘&#xff0c;而不是让你自甘堕落&#xff0c;脚下的路虽然难走&#xff0c;但我还能走&#xff0c;比起向阳而生&#xff0c;我更想尝试逆风…

Vite更新依赖缓存失败,强制更新依赖缓存

使用vitets开发一段时间了&#xff0c;感觉并不是想象中的好用&#xff0c;特别是出现些稀奇古怪的问题不好解决&#xff0c;比如下面这个问题 上午9:50:08 [vite] error while updating dependencies: Error: ENOENT: no such file or directory, open E:/workspace-dir/node…

Leetcode33 搜索旋转排序数组

题解&#xff1a; /*** 旋转排序数组可分为N1 N2两个部分&#xff0c;如&#xff1a;[4,5,6,7,1,2,3]&#xff0c;N1为[4,5,6,7]&#xff0c;N2为[1,2,3]** 必然满足以下两个条件&#xff1a;* 1. N1和N2都是分别递增的&#xff1b;* 2. N1中的所有元素大于N2中的所有元素;** …

研发协同工具哪个好用?比较常用的研发协同工具及其特点

Zoho Projects是一款在线的SaaS研发协同工具&#xff0c;支持敏捷开发/DevOps/Scrum等项目协作&#xff0c;最大的特点就是“会说话”&#xff0c;意思是&#xff1a;它可以把在项目协作过程中重要和相关的消息和信息通过恰到好处的方式告诉你&#xff0c;解决&#xff1a;开发…

Mac OS minicom 无法设置921600问题

MacOS minicom 无法设置921600问题 介绍过程解决方案参考资料 介绍 minicom是Mac上一款非常好用的串口工具。本文假设你已经安装minicom&#xff0c;并且知道minicom的一般配置和使用方法。这是“MacOS minicom 无法设置921600”的解决问题记录。它在以下环境中设置成功&#…

HarmonyOS NEXT新能力,一站式高效开发HarmonyOS应用

2023年8月6日华为开发者大会2023&#xff08;HDC.Together&#xff09;圆满收官&#xff0c;伴随着HarmonyOS 4的发布&#xff0c;华为向开发者发布了汇聚所有最新开发能力的HarmonyOS NEXT开发者预览版&#xff0c;并分享了围绕“一次开发&#xff0c;多端部署” “可分可合&a…

安防监控视频云存储平台EasyCVRH.265转码功能更新:新增分辨率配置

安防视频集中存储EasyCVR视频监控综合管理平台可以根据不同的场景需求&#xff0c;让平台在内网、专网、VPN、广域网、互联网等各种环境下进行音视频的采集、接入与多端分发。在视频能力上&#xff0c;视频云存储平台EasyCVR可实现视频实时直播、云端录像、视频云存储、视频存储…

docker compose部署zookeeper

单机部署 新建docker-compose.yaml version: 3 services:zookeeper:image: zookeeper:3.5.7container_name: base-zookeeperhostname: zookeeperprivileged: truerestart: alwaysports:- 2181:2181environment:TZ: "Asia/Shanghai"volumes:- ./volumes/zookeeper/d…

Python 处理 Excel 表格的 14 个常用操作

目录 1. 安装依赖库 2. 导入库 3. 读取Excel文件 4. 写入Excel文件 5. 创建工作表 6. 访问工作表 7. 读取单元格数据 8. 写入单元格数据 9. 获取行数和列数 10. 过滤数据 11. 排序数据 12. 添加新行 13. 删除行或列 14. 计算汇总统计 总结 无论是数据分析师、财…

实习笔记(一)

自定义注解&#xff1a; 自定义注解中有三个元注解Target,Retention,Document /*** 系统日志注解** author Mark sunlightcsgmail.com*/ Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) Documented public interface SysLog {String value() default "…

『C语言初阶』第八章 -隐式类型转换规则

前言 今天小羊又来给铁汁们分享关于C语言的隐式类型转换规则&#xff0c;在C语言中类型转换方式可分为隐式类型转换和显式类型转换(强制类型转换)&#xff0c;其中隐式类型转换是由编译器自动进行&#xff0c;无需程序员干预&#xff0c;今天小羊课堂说的就是关于隐式类型转换…

Splashtop 获得 ISO 27001 认证

2023年8月15日 加利福尼亚州库比蒂诺 Splashtop 在随处办公远程解决方案领域处于领先地位&#xff0c;该公司今天宣布已获得 ISO/IEC 27001 认证&#xff0c;印证了其对坚持最高标准的信息安全管理系统&#xff08;ISMS&#xff09;、为客户提供数据保护以及遵守法律法规要求的…

ChatGPT​保密吗?它有哪些潜在风险?如何规避?

自2022年11月公开发布以来&#xff0c;ChatGPT已成为许多企业和个人的必备工具&#xff0c;但随着该技术越来越多地融入我们的日常生活&#xff0c;人们很自然地想知道&#xff1a;ChatGPT是否是保密的。 问&#xff1a;ChatGPT保密吗&#xff1f; 答&#xff1a;否&#xff0…