c++ | vector

news2024/11/15 11:53:55

 前言

本篇博客讲解c++STL中的vector

💓 个人主页:普通young man-CSDN博客

⏩ 文章专栏:C++_普通young man的博客-CSDN博客

⏩ 本人giee:   普通小青年 (pu-tong-young-man) - Gitee.com

      若有问题 评论区见📝

🎉欢迎大家点赞👍收藏⭐文章

目录

vector的介绍

vector的使用

构造函数

赋值operator        

迭代器

容量

创建二维数组

1. 创建一维向量 v:

2. 创建二维向量 vv:

3. 修改元素:

4. 遍历二维向量:

vector实现

命名空间定义

类模板定义

公共类型定义

构造函数

析构函数

成员函数

测试函数


这篇博客我讲的不会很多,因为他和前面的string非常的像,并且更加简洁,还是一样我们根据文章来讲解,然后给一些特殊情况,然后再简单的实现一下vector.

vector的介绍

vector - C++ Reference (cplusplus.com)icon-default.png?t=N7T8https://legacy.cplusplus.com/reference/vector/vector/?kw=vector这个是一个c++的文档,可以根据这里面的介绍来合理的运用接口

vector的使用

构造函数

我写一下这些构造,我相信大家其实也可以看懂

#include <iostream>
#include <vector>

void test1() {
    // 构造一个包含10个元素的向量,所有元素初始化为0。
    std::vector<int> v(10, 0);

    // 复制构造 - 创建一个新的向量,包含与 'v' 相同的元素。
    std::vector<int> v1(v);

    // 迭代器构造 - 从 'v1' 的后五个元素创建一个新的向量。
    std::vector<int> v2(v1.begin() + 5, v1.end());

    // 初始化一个数组,并从其内容构建一个向量。
    int arr[10] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 0};
    std::vector<int> v3(arr, arr + sizeof(arr) / sizeof(arr[0]));

    // 遍历 'v3' 并打印其元素。
    for (int ch : v3) {
        std::cout << ch << " ";
    }
    std::cout << std::endl; // 打印换行符。
}

// 示例用法:
int main() {
    test1();
    return 0;
}
  1. 初始化:创建了一个名为 v 的 std::vector<int>,其中包含10个元素,每个元素都初始化为0。
  2. 复制构造:通过复制 v 的内容创建了一个新的向量 v1
  3. 子范围构造:使用 v1 的最后五个元素创建了一个新的向量 v2
  4. 数组到向量:声明并初始化了一个整数数组 arr,然后从该数组的元素构建了一个向量 v3
  5. 迭代和打印:使用范围for循环遍历 v3 中的元素,并将它们打印到控制台。

赋值operator        

#include <iostream>
#include <vector>

void test2() {
    // 创建两个向量,v1 和 v2,分别包含5个元素,初始化为1和2。
    std::vector<int> v1(5, 1);
    std::vector<int> v2(5, 2);

    // 将 v2 的内容赋给 v1。
    v1 = v2;

    // 打印 v1 的内容。
    std::cout << "v1: ";
    for (auto ch : v1) {
        std::cout << ch << " ";
    }
    std::cout << std::endl;

    // 再次将 v2 的内容赋给 v1(实际上这里没有改变 v1 的内容)。
    v1 = v2;

    // 打印 v2 的内容。
    std::cout << "v2: ";
    for (auto ch : v2) {
        std::cout << ch << " ";
    }
    std::cout << std::endl;
}

// 示例用法:
int main() {
    test2();
    return 0;
}
  1. 初始化向量:创建了两个 std::vector<int>v1v2,分别包含了5个元素,其中 v1 的所有元素被初始化为1,而 v2 的所有元素被初始化为2。

  2. 赋值操作:使用赋值操作符 (=) 将 v2 的内容赋给 v1,这意味着 v1 现在也包含了5个元素,每个元素的值都是2。

  3. 打印内容:使用范围for循环遍历 v1 并打印其内容。因为 v1 已经被赋值为 v2 的内容,所以输出将会是5个数字2。

  4. 再次赋值操作:再次将 v2 的内容赋给 v1,这实际上没有改变 v1 的内容,因为它已经与 v2 的内容相同了。

  5. 打印第二个向量的内容:使用范围for循环遍历 v2 并打印其内容。输出也将是5个数字2。

  6. 结束程序:函数 main 返回0,表示程序正常结束。

迭代器

#include <iostream>
#include <vector>

void test3() {
    // 初始化数组:定义了一个整数数组 arr,并初始化了10个元素。
    int arr[10] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 0};

    // 构造向量:使用数组的内容创建了一个常量 std::vector<int>,命名为 v。
    const std::vector<int> v(arr, arr + sizeof(arr) / sizeof(arr[0]));

    // 使用迭代器遍历向量:
    // 定义了一个 std::vector<int>::const_iterator 类型的迭代器 it,并将其初始化为指向向量的第一个元素。
    // 使用 while 循环遍历整个向量,每次迭代输出当前迭代器所指的元素,并递增迭代器。
    // 输出完所有元素后,换行。
    for (std::vector<int>::const_iterator it = v.begin(); it != v.end(); ++it) {
        std::cout << *it << " ";
    }
    std::cout << std::endl;

    // 使用范围for循环遍历向量:
    // 使用范围for循环遍历向量 v,并输出每个元素。
    for (auto ch : v) {
        std::cout << ch << " ";
    }
    std::cout << std::endl;

    // 使用反向迭代器遍历向量:
    // 定义了一个 std::vector<int>::const_reverse_iterator 类型的反向迭代器 it,并将其初始化为指向向量的最后一个元素。
    // 使用 while 循环遍历整个向量,每次迭代输出当前反向迭代器所指的元素,并递增反向迭代器。
    // 输出完所有元素后,换行。
    for (std::vector<int>::const_reverse_iterator it = v.rbegin(); it != v.rend(); ++it) {
        std::cout << *it << " ";
    }
    std::cout << std::endl;

    // 使用引用范围for循环遍历向量:
    // 使用范围for循环遍历向量 v,并输出每个元素的引用。虽然这里没有修改元素的值,但是这种方式可以用来修改向量中的元素。
    for (auto &ch : v) {
        std::cout << ch << " ";
    }
    std::cout << std::endl;
}

// 示例用法:
int main() {
    test3();
    return 0;
}
  1. 初始化数组

    • 定义了一个整数数组 arr,并初始化了10个元素。
  2. 构造向量

    • 使用数组的内容创建了一个常量 std::vector<int>,命名为 v
  3. 使用迭代器遍历向量

    • 定义了一个 std::vector<int>::const_iterator 类型的迭代器 it,并将其初始化为指向向量的第一个元素。
    • 使用 while 循环遍历整个向量,每次迭代输出当前迭代器所指的元素,并递增迭代器。
    • 输出完所有元素后,换行。
  4. 使用范围for循环遍历向量

    • 使用范围for循环遍历向量 v,并输出每个元素。
  5. 使用反向迭代器遍历向量

    • 定义了一个 std::vector<int>::const_reverse_iterator 类型的反向迭代器 it,并将其初始化为指向向量的最后一个元素。
    • 使用 while 循环遍历整个向量,每次迭代输出当前反向迭代器所指的元素,并递增反向迭代器。
    • 输出完所有元素后,换行。
  6. 使用引用范围for循环遍历向量

    • 使用范围for循环遍历向量 v,并输出每个元素的引用。虽然这里没有修改元素的值,但是这种方式可以用来修改向量中的元素。

        vector 迭代器失效问题。(重点) 迭代器的主要作用就是让算法能够不用关心底层数据结构,其底层实际就是一个指针,或者是对 指针进行了封装,比如:vector的迭代器就是原生态指针T* 。因此迭代器失效,实际就是迭代器 底层对应指针所指向的空间被销毁了,而使用一块已经被释放的空间,造成的后果是程序崩溃(即 如果继续使用已经失效的迭代器,程序可能会崩溃)。

这里我举几个例子

#include <iostream>
using namespace std;
#include <vector>
int main()
{
    vector<int> v{1,2,3,4,5,6};
    
    auto it = v.begin();
    
    // 将有效元素个数增加到100个,多出的位置使用8填充,操作期间底层会扩容
    // v.resize(100, 8);
    
    // reserve的作用就是改变扩容大小但不改变有效元素个数,操作期间可能会引起底层容
量改变
    // v.reserve(100);
    
    // 插入元素期间,可能会引起扩容,而导致原空间被释放
    // v.insert(v.begin(), 0);
    // v.push_back(8);
    
    // 给vector重新赋值,可能会引起底层容量改变
    v.assign(100, 8);
    
    /*
    出错原因:以上操作,都有可能会导致vector扩容,也就是说vector底层原理旧空间被释
放掉,而在打印时,it还使用的是释放之间的旧空间,在对it迭代器操作时,实际操作的是一块
已经被释放的空间,而引起代码运行时崩溃。
    解决方式:在以上操作完成之后,如果想要继续通过迭代器操作vector中的元素,只需给
it重新赋值即可。
    */
    while(it != v.end())
   {
        cout<< *it << " " ;
        ++it;
   }
    cout<<endl;
    return 0;
}

选择了v.assign(100, 8);这一行来进行操作。这行代码将v的内容替换为100个值为8的元素。由于v的大小从6个元素增加到了100个元素,这很可能导致std::vector需要重新分配内存以容纳更多的元素。在重新分配过程中,原有的内存会被释放,并且std::vector会在新的内存位置上构造新元素。这意味着之前的迭代器it指向的地址不再有效。

#include <iostream>e
/*rase删除pos位置元素后,pos位置之后的元素会往前搬移,没有导致底层空间的改变,理
论上讲迭代器不应该会失效,但是:如果pos刚好是最后一个元素,删完之后pos刚好是end
的位置,而end位置是没有元素的,那么pos就失效了。因此删除vector中任意位置上元素
时,vs就认为该位置迭代器失效了。
以下代码的功能是删除vector中所有的偶数,请问那个代码是正确的,为什么?*/
using namespace std;
#include <vector>
int main()
{
 int a[] = { 1, 2, 3, 4 };
 vector<int> v(a, a + sizeof(a) / sizeof(int));
 // 使用find查找3所在位置的iterator
 vector<int>::iterator pos = find(v.begin(), v.end(), 3);
 // 删除pos位置的数据,导致pos迭代器失效。
 v.erase(pos);
 cout << *pos << endl; // 此处会导致非法访问
 return 0;
}
  1. v.erase(pos); 删除了pos所指向的元素。按照题目描述,这段代码的目标是删除所有偶数,但这里只删除了一个元素(即值为3的元素),这与题目要求不符。

  2. 在删除一个元素后,pos迭代器确实可能失效。虽然erase操作不会导致底层内存重新分配,但如果删除的是pos指向的元素,那么pos将不再指向任何有效元素。特别是,如果pos原本指向的是最后一个元素,那么删除后它将指向v.end(),这是无效的。

  3. 删除元素后尝试访问pos所指向的元素会导致未定义行为,因为pos已经失效。

还有和这个删除是一样的就是头插入,这个也会出现迭代器失效,因为pos位置没有进行映射改变


容量

#include <iostream>
#include <vector>

void test4() {
    // 初始化一个整数数组。
    int arr[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 0};

    // 使用数组的内容构造一个 std::vector<int>,命名为 v。
    std::vector<int> v(arr, arr + sizeof(arr) / sizeof(arr[0]));

    // 输出向量的信息。
    std::cout << "大小 -> " << v.size() << std::endl;
    std::cout << "容量 -> " << v.capacity() << std::endl;

    // 调整向量的大小,并填充额外的元素。
    v.resize(12, 1);
    for (auto& ch : v) {
        std::cout << ch << " ";
    }
    std::cout << std::endl;

    // 再次调整向量的大小,并填充额外的元素。
    v.resize(5, 2);
    for (auto& ch : v) {
        std::cout << ch << " ";
    }

    // 创建一个向量,并移除其中的偶数元素。
    std::vector<int> v1{1, 2, 3, 4, 5, 6};
    auto it = v1.begin();
    while (it != v1.end()) {
        if (*it % 2 == 0) {
            it = v1.erase(it); // 注意:erase 返回指向下一个元素的新迭代器。
        } else {
            ++it;
        }
    }
    for (auto e : v1) {
        std::cout << e << " ";
    }
    std::cout << std::endl;
}

// 示例用法:
int main() {
    test4();
    return 0;
}

下面是代码中使用的 std::vector 接口的详细解释:

  1. 构造函数

    1std::vector<int> v(arr, arr + sizeof(arr) / sizeof(arr[0]));
    • 这里使用了 std::vector 的构造函数,该构造函数接受两个迭代器参数,用于指定数组的开始和结束位置。
    • arr 是数组的起始地址,arr + sizeof(arr) / sizeof(arr[0]) 是数组的结束地址。
  2. 输出向量信息

    1std::cout << "大小 -> " << v.size() << std::endl;
    2std::cout << "容量 -> " << v.capacity() << std::endl;
    • size() 函数返回向量中元素的数量。
    • capacity() 函数返回向量当前分配的存储空间,它可以大于向量的实际大小。
  3. 调整向量大小

    1v.resize(12, 1);
    • resize(n, val) 函数用于调整向量的大小。如果新大小大于当前大小,则在向量末尾添加 val 值填充的新元素;如果新大小小于当前大小,则截断向量。
  4. 遍历向量

    1for (auto& ch : v) {
    2    std::cout << ch << " ";
    3}
    • 使用范围for循环遍历向量 v,并输出每个元素。
  5. 移除向量中的偶数元素

    1auto it = v1.begin();
    2while (it != v1.end()) {
    3    if (*it % 2 == 0) {
    4        it = v1.erase(it); // 注意:erase 返回指向下一个元素的新迭代器。
    5    } else {
    6        ++it;
    7    }
    8}
    • 使用迭代器 it 遍历向量 v1
    • 当遇到偶数元素时,使用 erase(it) 函数删除该元素,并将迭代器更新为指向下一个元素的新迭代器。
    • 如果元素不是偶数,则递增迭代器。

        总结来说,这段代码展示了如何使用 std::vector 的构造函数、size()capacity()resize() 函数,以及如何遍历和修改向量中的元素。特别是,在删除向量中的元素时,需要注意 erase() 函数返回的是指向下一个元素的新迭代器,以避免潜在的迭代器失效问题。


创建二维数组

void test5(){
    vector<int> v(5,0);  
    vector<vector<int>>vv(5,v);
    vv[0][1] = 5;//operator[]返回
    //第一个[],访问的是外层的vector -- 也就是行
    //第二个[],访问的是里层的vector  -- 也就是列
    //遍历二维数组
    for (size_t i = 0; i < vv.size(); i++)
    {
        for (size_t j = 0; j < v.size();j++) {
            cout << vv[i][j] << " ";
        }
        cout << endl;
    }
}
1. 创建一维向量 v
  • vector<int> v(5,0); 创建了一个包含5个整数0的一维向量 v。这里的构造函数 vector(size_t n, const T val = T()) 被调用,其中 n 是5,val 是0。
2. 创建二维向量 vv
  • vector<vector<int>> vv(5, v); 创建了一个包含5个一维向量 v 的二维向量 vv。这里也是使用了构造函数 vector(size_t n, const T val = T()),其中 n 是5,val 是之前创建的一维向量 v
3. 修改元素:
  • vv[0][1] = 5; 这行代码修改了二维向量 vv 中第一行(索引0)第二列(索引1)的元素值为5。
4. 遍历二维向量:
  • for (size_t i = 0; i < vv.size(); i++):外层循环用于遍历二维向量 vv 的每一行。
  • for (size_t j = 0; j < v.size(); j++):内层循环用于遍历每一行中的每一个元素。
  • cout << vv[i][j] << " ";:输出当前元素的值,并用空格分隔。
  • cout << endl;:每行输出完毕后换行。


vector实现

这个实现我只会实现个别的接口

#pragma once
#include <iostream>
#include <vector>
#include <cassert>
using namespace std;

namespace yang {
    // 定义类模板
    template <class T>
    class vector {
    public:
        typedef T* iterator;          // 非常量迭代器类型定义
        typedef const T* const_iterator; // 常量迭代器类型定义

        // 默认构造函数
        vector() = default;

        // 拷贝构造函数
        vector(const vector<T>& v) {
            // 分配与原向量相同大小的空间
            reserve(v.size());
            for (const auto& it : v) {
                pushback(it); // 将原向量的每个元素复制到新向量
            }
        }

        // 迭代器范围构造函数
        template <class inputiterator>
        vector(inputiterator fast, inputiterator last) {
            while (fast != last) {
                pushback(*fast); // 将迭代器范围内的元素添加到向量中
                ++fast;
            }
        }

        // 构造函数,创建指定大小的向量,并用给定值初始化
        vector(size_t n, const T val = T()) {
            reserve(n); // 分配足够大的空间
            for (int i = 0; i < n; i++) {
                pushback(val); // 添加初始化值
            }
        }

        // 构造函数,创建指定大小的向量,并用给定值初始化
        // 注意:使用 int 参数可能会导致溢出问题
        vector(int n, const T val = T()) {
            reserve(n); // 分配足够大的空间
            for (int i = 0; i < n; i++) {
                pushback(val); // 添加初始化值
            }
        }

        // 析构函数
        ~vector() {
            delete[] _start; // 释放分配的内存
            _start = _finish = _end_of_storage = nullptr; // 清理指针
        }

        // 获取向量中的元素数量
        size_t size() const {
            return _finish - _start;
        }

        // 获取向量的容量
        size_t capacity() const {
            return _end_of_storage - _start;
        }

        // 判断向量是否为空
        bool empty() {
            return _finish == _start;
        }

        // 清空向量内容
        void clear() {
            _finish = _start;
        }

        // 返回向量开始迭代器
        iterator begin() {
            return _start;
        }

        // 返回向量开始的常量迭代器
        const_iterator begin() const {
            return _start;
        }

        // 返回向量结束迭代器
        iterator end() {
            return _finish;
        }

        // 返回向量结束的常量迭代器
        const_iterator end() const {
            return _finish;
        }

        // 扩容函数
        void reserve(size_t n) {
            if (n > capacity()) {
                // 记录原来的元素数量
                size_t old_size = size();

                // 分配新的内存空间
                T* tmp = new T[n];
                //memcpy(tmp, _start, sizeof(T) * size()); // 旧版代码使用memcpy
                for (int i = 0; i < old_size; i++) {
                    tmp[i] = _start[i]; // 复制原有元素
                }

                // 销毁旧内存
                delete[] _start;

                // 更新指针
                _start = tmp;
                _finish = tmp + old_size;
                _end_of_storage = tmp + n;
            }
        }

        // 调整向量大小
        void resize(size_t n, T val = T()) {
            if (n < size()) {
                _finish = _start + n; // 缩小向量
            } else {
                reserve(n); // 扩容
                while (_finish < _start + n) {
                    *_finish = val; // 填充新元素
                    ++_finish;
                }
            }
        }

        // 在向量末尾添加元素
        void pushback(const T& val) {
            if (_finish == _end_of_storage) {
                reserve(capacity() == 0 ? 4 : capacity() * 2); // 如果空间不足则扩容
            }
            *_finish = val; // 添加新元素
            ++_finish;
        }

        // 移除向量末尾的元素
        void popback() {
            assert(!empty()); // 断言向量非空
            --_finish; // 向量末尾减一
        }

        // 交换两个向量的内容
        void swap(vector<T>& v) {
            std::swap(_start, v._start);
            std::swap(_finish, v._finish);
            std::swap(_end_of_storage, v._end_of_storage);
        }

        // 赋值运算符重载
        vector<T>& operator=(vector<T> v) {
            swap(v); // 使用swap函数完成赋值
            return *this;
        }

        // 下标访问
        T& operator[](size_t i) {
            assert(i < size()); // 断言索引在范围内
            return _start[i]; // 返回元素的引用
        }

        // 常量下标访问
        const T& operator[](size_t i) const {
            assert(i < size()); // 断言索引在范围内
            return _start[i];
        }

        // 在指定位置插入元素
        iterator insert(iterator pos, const T& val) {
            assert(pos >= _start && pos < _finish); // 断言位置合法
            if (_finish == _end_of_storage) {
                // 存储pos到_start的位置(如果扩容,地址会改变)
                size_t len = pos - _start;

                // 扩容
                reserve(capacity() == 0 ? 4 : capacity() * 2);
                // 更新位置(映射)
                pos = _start + len;
            }

            // 向右移动元素以留出空间
            iterator end = _finish - 1;
            while (end >= pos) {
                *(end + 1) = *end;
                --end;
            }

            *pos = val; // 插入新元素
            ++_finish; // 更新结束指针

            // 返回新元素的位置(注意返回pos+1是为了不影响实参)
            return pos + 1;
        }

        // 删除指定位置的元素
        iterator erase(iterator pos) {
            assert(pos >= _start && pos < _finish); // 断言位置合法

            // 移动后面的元素向前覆盖被删除的元素
            iterator it = pos + 1;
            while (it != _finish) {
                *(it - 1) = *it;
                ++it;
            }

            --_finish; // 更新结束指针
            return pos; // 返回删除位置
        }

    private:
        iterator _start = nullptr; // 开始迭代器
        iterator _finish = nullptr; // 结束迭代器
        iterator _end_of_storage = nullptr; // 存储区结束迭代器
    };

    //-- 测试函数 --
    void test1() {
        vector<int> v;
        v.reserve(5); // 预留5个元素的空间
        v.pushback(1);
        v.pushback(2);
        v.pushback(3);
        v.pushback(4);
        v.pushback(5);
        v.pushback(1);
        v.pushback(2);
        v.pushback(3);
        v.pushback(4);
        v.pushback(5);

        cout << v.size() << endl; // 输出向量的大小
        cout << v.capacity() << endl; // 输出向量的容量
        auto pos = find(v.begin(), v.end(), 2); // 查找值2的位置
        pos = v.insert(pos, 10); // 在找到的位置插入10
        *pos *= 10; // 修改插入的元素值为100

        // 利用迭代器打印
        vector<int>::iterator it = v.begin();
        while (it != v.end()) {
            cout << *it << " ";
            ++it;
        }
    }

    void test2() {
        // 删除
        vector<int> v;
        for (int i = 0; i < 10; i++) {
            v.pushback(i); // 添加偶数
            v.pushback(i); // 添加偶数
        }

        auto it = v.begin(); // 自动类型推导
        while (it != v.end()) {
            if (*it % 2 == 0) {
                it = v.erase(it); // 删除偶数
            } else {
                ++it;
            }
        }

        for (auto ch : v) {
            cout << ch << " "; // 输出剩余的奇数
        }
    }

    void test3() {
        vector<int> v;
        for (int i = 0; i < 10; i++) {
            v.pushback(i); // 添加0-9
        }

        cout << v.size() << endl; // 输出向量大小
        for (auto ch : v) {
            cout << ch << " "; // 输出所有元素
        }
        cout << endl;

        v.resize(5); // 缩小向量
        for (auto ch : v) {
            cout << ch << " "; // 输出前五个元素
        }
        cout << endl;

        v.resize(20); // 扩大向量
        for (auto ch : v) {
            cout << ch << " "; // 输出所有元素
        }
    }

    void test4() {
        vector<int> a1;
        a1.pushback(1);
        a1.pushback(1);
        a1.pushback(1);
        a1.pushback(1);
        a1.pushback(1);

        vector<int> a2 = a1; // 拷贝构造
        for (auto ch : a2) {
            cout << ch << " "; // 输出拷贝的内容
        }
        cout << endl;

        a1.clear(); // 清空a1
        cout << a1.size() << endl; // 输出a1的大小
    }

    void test5() {
        vector<string> v;
        v.pushback("hello"); // 添加字符串"hello"
        vector<string> vv(v); // 拷贝构造
        for (auto ch : vv) {
            cout << ch << " "; // 输出拷贝的内容
        }
    }
}

命名空间定义

  • namespace yang { ... }: 定义了一个命名空间 yang,用来避免与其他部分的程序或标准库中的名字发生冲突。

类模板定义

  • template<class T> class vector { ... }: 定义了一个名为 vector 的类模板,它可以为任何数据类型 T 实例化。

公共类型定义

  • typedef T* iterator;: 定义了一个非常量迭代器的类型别名。
  • typedef const T* const_iterator;: 定义了一个常量迭代器的类型别名。

构造函数

  • vector() = default;: 默认构造函数,使用编译器提供的默认行为。
  • vector(const vector<T>& v);: 拷贝构造函数,用于通过拷贝另一个 vector 创建新的 vector
  • template <class InputIterator> vector(InputIterator first, InputIterator last);: 范围构造函数,接受两个迭代器,将它们之间的元素复制到新的 vector 中,并以此初始化 vector
  • vector(size_t n, const T val = T());: 构造函数,创建一个含有 n 个元素的 vector,这些元素都被初始化为 val
  • vector(int n, const T val = T());: 类似于上述构造函数,但参数是 int 类型。这种写法通常不推荐,因为 int 可能会导致溢出问题。

析构函数

  • ~vector();: 析构函数,释放 vector 占用的内存。

成员函数

  • size_t size() const;: 返回 vector 中的元素数量。
  • size_t capacity() const;: 返回 vector 的容量,即已分配的存储空间可以容纳的元素数量。
  • bool empty();: 如果 vector 是空的,则返回 true
  • void clear();: 清除 vector 中的所有元素,但保留其分配的存储空间。
  • iterator begin();: 返回指向 vector 第一个元素的迭代器。
  • const_iterator begin() const;: 返回指向 vector 第一个元素的常量迭代器。
  • iterator end();: 返回指向 vector 末尾之后位置的迭代器。
  • const_iterator end() const;: 返回指向 vector 末尾之后位置的常量迭代器。
  • void reserve(size_t n);: 预先分配足够存放 n 个元素的存储空间,但不一定改变 vector 的大小。

memcpy是内存的二进制格式拷贝,将一段内存空间中内容原封不动的拷贝到另外一段内存 空间中

 如果拷贝的是自定义类型的元素,memcpy既高效又不会出错,但如果拷贝的是自定义类型 元素,并且自定义类型元素中涉及到资源管理时,就会出错,因为memcpy的拷贝实际是浅 拷贝。

  • void resize(size_t n, T val = T());: 改变 vector 的大小为 n,并用 val 初始化新添加的元素。
  • void push_back(const T& val);: 在 vector 的末尾添加一个新元素。
  • void pop_back();: 删除 vector 末尾的一个元素。
  • void swap(vector<T>& v);: 交换两个 vector 的内容。
  • vector<T>& operator=(vector<T> v);: 移动赋值运算符,用于从另一个 vector 复制内容到当前 vector
  • T& operator[](size_t i);: 返回对 vector 中第 i 个元素的引用。
  • const T& operator[](size_t i) const;: 返回对 vector 中第 i 个元素的常量引用。
  • iterator insert(iterator pos, const T& val);: 在给定的迭代器位置 pos 插入一个值 val
  • iterator erase(iterator pos);: 删除给定迭代器位置 pos 指向的元素。

测试函数

  • test1(): 测试 vector 的基本功能,如 reservepush_backinsert, 和 find
  • test2(): 测试 erase 函数来删除满足特定条件的元素。
  • test3(): 测试 resize 函数改变 vector 的大小。
  • test4(): 测试拷贝构造函数和 clear 函数。
  • test5(): 测试字符串类型的 vector 拷贝构造函数。

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

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

相关文章

珠江电缆,顺应全球变化,实现高质量出海

在全球经济快速变化的今天&#xff0c;越来越多的企业将目光投向了国际市场。特别是对于线缆行业来说&#xff0c;顺应全球变化、应对机遇与挑战&#xff0c;实现高质量出海已成为长期发展的战略目标之一。珠江电缆作为一家集研发、制造和销售为一体的大型专业电线电缆企业&…

智能电表在什么情况下需要加装互感器?

智能电表作为现代电力系统中不可或缺的一部分&#xff0c;负责准确计量电力消耗。然而&#xff0c;在某些特定条件下&#xff0c;仅凭智能电表无法满足高精度测量需求&#xff0c;此时便需引入互感器。本文将深入解析智能电表与互感器的协作原理&#xff0c;明确指出加装互感器…

ASR(Automatic Speech Recognition)调研,当前sota的架构

asr概览英文纯享版&#xff1a;an overview of transducer models for asr 本文主要讲述nvidia和openai的模型架构&#xff0c;应为他们两家霸榜huggingface leader board 小白也能阅读了解一下当前sota的asr架构是什么样的 评测指标 Word Error Rate (WER)&#xff1a;错词率…

关于使用QListView向模型插入第一条数据的相关问题

在使用QListView的时候&#xff0c;看着它的四个尖角不爽&#xff0c;想着让它变成圆角矩形吗&#xff0c;但麻烦就接踵而至了。。。 在向其中插入第一条数据的时候发现&#xff0c;你插入的一条数据它存在边界超过QListView的现象。如下图所示&#xff1a; 这就令人头大了&am…

JavaEE 图书管理系统

基于阿里巴巴的fastjson框架搭建的JavaEE版本的图书管理系统&#xff0c;项目架构如下&#xff1a; fastjson包的阿里云下载镜像如下&#xff1a; Central Repository: com/alibaba/fastjson2/fastjson2/2.0.8 运行效果&#xff1a; Bean Book.java package Bean;public c…

2024华数杯大学生数学建模竞赛(C题)数学建模完整思路+完整代码全解全析

你是否在寻找数学建模比赛的突破点&#xff1f;数学建模进阶思路&#xff01; 作为经验丰富的数学建模团队&#xff0c;我们将为你带来2024华数杯数学建模竞赛&#xff08;C题&#xff09;的全面解析。这个解决方案包不仅包括完整的代码实现&#xff0c;还有详尽的建模过程和解…

按照指定格式打印pprint()

【小白从小学Python、C、Java】 【考研初试复试毕业设计】 【Python基础AI数据分析】 按照指定格式打印 pprint() [太阳]选择题 根据给定的Python代码&#xff0c;哪个选项是正确的&#xff1f; from pprint import pprint data { name: A, age: 30, hobbies:…

Vue项目学习(项目的开发流程)(2)

1、vue项目的默认首页和入口文件 2、两种书写的方式是表达一样的意思——>el&#xff1a;指定当前Vue实例挂载到哪个区域里 3、如果属性值和属性名一致&#xff0c;冒号和后面可以省略不写 &#xff08;所以有两种写法&#xff09; 4、以".vue"文件结尾的文件在项…

3个步骤上⼿Midjourney表情包教程,并上传到微信实现变现!

羡慕别⼈设计的表情包,有趣⼜好玩~也想拥有⾃⼰的个性表情包,可是⾯对复杂的设计流程,却不知从何开始?现在⽤Midjourney,你就可以轻松制作,各种⻛格的表情包,变钱赚钱,这些⽅法分享给 你~ 通⽤公式: 我们⽤表情包魔法公式,加⼊你想要的风格,⽐如漫画、卡通、插画、…

rpc框架怎么使用

这是我们提供RPC的服务类&#xff1a; class MprpcApplication { public:static void Init(int argc, char **argv);static MprpcApplication& GetInstance();static MprpcConfig& GetConfig(); private:static MprpcConfig m_config;MprpcApplication(){}MprpcApplica…

cuda逐步优化实现reduce sum 操作

归约是一种常见的数据并行原语&#xff0c;它将数组中的元素通过某种二元操作&#xff08;如加法&#xff09;合并成一个单一的值。通过逐步展示不同的CUDA实现版本&#xff0c;来演示重要的优化策略。 由于规约的算术操作很简单&#xff0c;对算力要求不高&#xff0c;因此我们…

文件上传和下载

要想实现文件上传和下载&#xff0c;其实只需要下述代码即可&#xff1a; 文件上传和下载 import cn.hutool.core.io.FileUtil; import cn.hutool.core.util.StrUtil; import com.example.common.Result; import org.springframework.web.bind.annotation.*; import org.sprin…

SQL注入sqli-labs-master关卡二

第二关如下&#xff1a; 查看页面与第一关差不多的样子&#xff0c;再查看PHP源码&#xff0c;与第一关差不多只是其中的查询处有不同。&#xff08;查看源码是为了更好的判断出该页面有什么漏洞&#xff09;其中没有单引号所以不用添加单引号去闭合去逃离单引号&#xff0c;说…

【生成式AI-一-生成式AI到底在说什么】

成式AI到底在说什么 什么是生成式人工智能生成式人工智能、机器学习、深度学习的关系chat-gpt 到底是如何实现对话的&#xff1f; 今天主要来看到底生成式AI是什么&#xff0c;语言模型是如何实现生成这个功能的&#xff1f; 什么是生成式人工智能 现在人工智能能做的事情很多…

pxe环境下的无人值守自动安装

0. 环境部署前的准备 1.rhel7的主机 2.开启主机图形 3.配置网络可用 4.关闭vmware dhcp功能 5.关闭防火墙以及selinux getenforce要为disable状态 grubby --update-kernel ALL --args selinux0 systemctl disable --now firewalld 1.kickstart自动安装脚本制作 我们想要…

SQL注入之webshell上传

首先webshell上传就是利用MySQL的文件读写注入而实现,要想上传webshell就得了解文件读写注入的原理。文件读写注入就是利用文件的读写权限进行注入&#xff0c;它可以写一句话木马&#xff0c;也可以读取文件系统的敏感信息。 文件读写注入的条件&#xff1a; 1.高版本的MYSQ…

Axure导入ElementUI元件库——提升原型设计效率与质量

在快速迭代的互联网产品开发过程中&#xff0c;高质量的原型设计是确保项目顺利进行的关键一步。Axure RP&#xff0c;作为一款强大的原型设计工具&#xff0c;以其丰富的交互功能和易用的界面设计&#xff0c;深受设计师和开发者的喜爱。而ElementUI&#xff0c;作为一套为开发…

Ubuntu配置carla docker环境

前言: 本文只在以下设备成功运行, 其他设备不保证能成功, 可以参考在自己设备进行配置 环境 ubuntu 20.04carla 0.9.15gpu 3060(notebook) 安装显卡驱动&nvidia-container-toolkit 显卡驱动 安装完成系统后直接在’软件和更新->附加驱动’直接选择470(proprietary…

工程化实践:工程配置化设计

文内项目 Github&#xff1a;XIAOJUSURVEY 配置化是很灵活且很常见的使用&#xff0c;那XIAOJUSURVEY里有哪些地方应用到了呢&#xff1f; 基础模板​ 问卷模板​ 在创建问卷时&#xff0c;我们提供了多种问卷类型选择&#xff0c;例如普通问卷、投票、报名、NPS等。 为了实…

安卓碎片Fragment

文章目录 Fragment的简单用法动态添加Fragment在Fragment中实现返回栈碎片与活动之间的通信 Fragment是一种可以嵌入在Activity当中的UI片段&#xff0c;它能让程序更加合理和充分地利用大屏幕的空间。 Fragment的简单用法 先建一个空的项目&#xff0c;然后创建两个fragment文…