C++ 中的向量实现
本文将演示如何在 C++ 中实现一个类似于 std::vector
的类。
使用 malloc
和 realloc
函数在 C++ 中实现自定义 vector
类
std::vector
容器被称为动态 C 样式数组,它连续存储其元素。尽管没有强制执行确切的实现,但规范要求容器的某些功能。也就是说,向量应该是一个有序的数据结构,并提供对其元素的随机访问。其他功能可以表示为作为接口公开的公共成员函数,其中一些我们将在以下示例中实现。请注意,实际的 std::vector
实现可能会非常广泛,因此我们只是展示了读者可能会添加更多功能的起点。
首先,我们需要定义一个 MyVector
类作为可以存储任何泛型类型的函数模板。然后我们可以包含核心数据成员,如指向元素数组的指针和整数对象,以相应地存储大小/容量。请记住,为了获得更好的性能,std::vector
可能会分配比存储元素更大的内存块。分配的内存大小称为向量的容量,我们将其存储在 cap
数据成员中。一旦构建了新的 MyVector
对象并将当前元素设置为零,我们将分配固定数量(20
个对象)。
请注意,我们使用了 malloc
函数,这在当代 C++ 中可能被认为是有害的,但如果谨慎使用,它会提供灵活性和良好的性能。此外,使用 malloc
分配的内存可以使用 realloc
函数进行扩展,我们将需要它来管理数组的大小调整。
总的来说,我们的类包含三个成员函数和 []
运算符,它们构成了动态向量的基础。以下示例代码演示了 push_back
、size
和 operator[]
函数实现。后两者非常直观且易于理解,而 push_back
可能需要一些解释。
请注意
,当元素数量超过容量时,我们只需要在构造对象后分配新的内存区域。因此,对于push_back
函数中的每个场景,我们需要两个单独的路径,其中一个将调用realloc
函数来扩展当前内存块。realloc 接受两个参数:指向前一个内存区域的指针和新区域的大小(以字节为单位)。如果调用成功,则返回一个有效指针,与前一个指针相同或一个新指针,具体取决于在同一块中是否可以找到足够的内存。否则,返回 NULL 指针以表示请求失败。
#include <iostream>
#include <string>
using std::cout; using std::cin;
using std::endl; using std::string;
template<typename T>
class MyVector {
public:
MyVector() {
cap = alloc;
vector = (T*)malloc(sizeof(T) * alloc);
elem_num = 0;
};
void push_back(const T &data);
void pop_back();
[[nodiscard]] bool empty() const;
[[nodiscard]] size_t size() const;
[[nodiscard]] size_t capacity() const;
T &operator[](size_t pos);
~MyVector();
private:
T *vector = nullptr;
size_t cap;
size_t elem_num;
const int alloc = 20;
};
template<typename T>
MyVector<T>::~MyVector() {
free(vector);
}
template<typename T>
void MyVector<T>::push_back(const T &data) {
if (elem_num < cap) {
*(vector + elem_num) = data;
elem_num++;
} else {
vector = (T*)realloc(vector, sizeof(T) * cap * 2);
cap *= 2;
if (vector) {
*(vector + elem_num) = data;
elem_num++;
}
}
}
template<typename T>
void MyVector<T>::pop_back() {
if (empty())
return;
elem_num--;
}
template<typename T>
T &MyVector<T>::operator[](size_t pos) {
if (pos >= 0 && pos <= elems)
return *(this->arr + pos);
throw std::out_of_range("Out of bounds element access");
}
template<typename T>
size_t MyVector<T>::capacity() const {
return cap;
}
template<typename T>
bool MyVector<T>::empty() const {
return elem_num == 0;
}
template<typename T>
size_t MyVector<T>::size() const {
return elem_num;
}
struct MyClass {
int num;
double num2;
};
int main() {
MyVector<MyClass> m1;
m1.push_back({1, 1.1});
m1.push_back({1, 1.2});
m1.push_back({1, 1.3});
m1.push_back({1, 1.4});
for (size_t i = 0; i < m1.size(); ++i) {
cout << m1[i].num << ", " << m1[i].num2 << endl;
}
cout << "/ ------------------- /" << endl;
m1.pop_back();
m1.pop_back();
for (size_t i = 0; i < m1.size(); ++i) {
cout << m1[i].num << ", " << m1[i].num2 << endl;
}
return EXIT_SUCCESS;
}
输出:
1, 1.1
1, 1.2
1, 1.3
1, 1.4
/ ------------------- /
1, 1.1
1, 1.2
为了测试 MyVector
类,我们在 main
函数中包含了一些驱动程序代码,并定义了 MyClass
自定义对象以存储为元素。最后,我们添加了 pop_back
函数,用于删除向量后面的元素。pop_back
函数不需要释放或删除元素内容。它可以在每次调用时将 elem_num
成员减一,下一次 push_back
调用将重写丢弃的元素位置。一旦对象超出范围,释放内存块也很重要。因此,我们需要在类的析构函数中包含对 free
函数的调用。
#include <iostream>
#include <string>
using std::cout; using std::cin;
using std::endl; using std::string;
template<typename T>
class MyVector {
public:
MyVector() {
arr = new T[default_capacity];
cap = default_capacity;
elems = 0;
};
void push_back(const T &data);
void pop_back();
[[nodiscard]] bool empty() const;
[[nodiscard]] size_t size() const;
[[nodiscard]] size_t capacity() const;
T &operator[](size_t pos);
~MyVector();
private:
T *arr = nullptr;
size_t cap;
size_t elems;
const int default_capacity = 20;
};
template<typename T>
MyVector<T>::~MyVector() {
delete [] arr;
}
template<typename T>
void MyVector<T>::push_back(const T &data) {
if (elems < cap) {
*(arr + elems) = data;
elems++;
} else {
auto tmp_arr = new T[cap * 2];
cap *= 2;
for (int i = 0; i < elems; i++) {
tmp_arr[i] = arr[i];
}
delete [] arr;
arr = tmp_arr;
*(arr + elems) = data;
elems++;
}
}
template<typename T>
T &MyVector<T>::operator[](size_t pos) {
if (pos >= 0 && pos <= elems)
return *(this->arr + pos);
throw std::out_of_range("Out of bounds element access");
}
template<typename T>
size_t MyVector<T>::size() const {
return elems;
}
template<typename T>
size_t MyVector<T>::capacity() const {
return cap;
}
template<typename T>
void MyVector<T>::pop_back() {
if (empty())
return;
elems--;
}
template<typename T>
bool MyVector<T>::empty() const {
return elems == 0;
}
struct MyClass {
string name;
double num;
};
int main() {
MyVector<MyClass> m1;
m1.push_back({"one", 1.1});
m1.push_back({"two", 1.2});
m1.push_back({"three", 1.3});
m1.push_back({"four", 1.4});
for (size_t i = 0; i < m1.size(); ++i) {
cout << m1[i].name << ", " << m1[i].num << endl;
}
cout << "/ ------------------- /" << endl;
m1.pop_back();
m1.pop_back();
for (size_t i = 0; i < m1.size(); ++i) {
cout << m1[i].name << ", " << m1[i].num << endl;
}
return EXIT_SUCCESS;
}
输出:
one, 1.1
two, 1.2
three, 1.3
four, 1.4
/ ------------------- /
one, 1.1
two, 1.2
相关文章
在 C++ 中通过掷骰子生成随机值
发布时间:2023/04/09 浏览次数:169 分类:C++
-
本文解释了如何使用时间因子方法和模拟 C++ 中的掷骰子的任意数方法生成随机数。了解它是如何工作的以及它包含哪些缺点。提供了一个 C++ 程序来演示伪数生成器。
在 C++ 中使用模板的链表
发布时间:2023/04/09 浏览次数:158 分类:C++
-
本文解释了使用模板在 C++ 中创建链表所涉及的各个步骤。工作程序演示了一个链表,该链表使用模板来避免在创建新变量时声明数据类型的需要。
在 C++ 中添加定时延迟
发布时间:2023/04/09 浏览次数:142 分类:C++
-
本教程将为你提供有关在 C++ 程序中添加定时延迟的简要指南。这可以使用 C++ 库为我们提供的一些函数以多种方式完成。
在 C++ 中创建查找表
发布时间:2023/04/09 浏览次数:155 分类:C++
-
本文重点介绍如何创建查找表及其在不同场景中的用途。提供了三个代码示例以使理解更容易,并附有代码片段以详细了解代码。
如何在 C++ 中把字符串转换为小写
发布时间:2023/04/09 浏览次数:63 分类:C++
-
介绍了如何将 C++ std::string 转换为小写的方法。当我们在考虑 C++ 中的字符串转换方法时,首先要问自己的是我的输入字符串有什么样的编码
如何在 C++ 中确定一个字符串是否是数字
发布时间:2023/04/09 浏览次数:163 分类:C++
-
本文介绍了如何检查给定的 C++ 字符串是否是数字。在我们深入研究之前,需要注意的是,以下方法只与单字节字符串和十进制整数兼容。
如何在 c++ 中查找字符串中的子字符串
发布时间:2023/04/09 浏览次数:65 分类:C++
-
本文介绍了在 C++ 中检查一个字符串是否包含子字符串的多种方法。使用 find 方法在 C++ 中查找字符串中的子字符串
如何在 C++ 中把字符串转换为 Char 数组
发布时间:2023/04/09 浏览次数:107 分类:C++
-
本文介绍了在 C++ 中把字符串转换为 char 数组的多种方法。使用 std::basic_string::c_str 方法将字符串转换为 char 数组