C++STL序列式容器——vector容器详解

C++STL序列式容器——vector容器详解_第1张图片
纵有疾风起,人生不言弃。本文篇幅较长,如有错误请不吝赐教,感谢支持。

文章目录

    • 一.vector容器基本概念
    • 二.vector常用操作
      • ①vector构造函数
      • ②特性操作
      • ③元素操作
      • ④赋值操作
      • ⑤交换操作
      • ⑥比较操作
      • ⑦插入和删除操作

一.vector容器基本概念

vector的数据安排以及操作方式,与array非常相似,都是一片连续的储存空间,两者的唯一差别在于空间的运用的灵活性。

array是静态数组,一旦配置了空间就无法再改变,要换大一点或者小一点的空间,可以,一切琐碎得由自己来,首先配置一块新的空间,然后将旧空间的数据搬往新空间,再释放原来的空间。

vector是动态空间(动态数组),随着元素的加入,它的内部机制会自动扩充空间以容纳新元素。因此vector的运用对于内存的合理利用与运用的灵活性有很大的帮助,我们再也不必害怕空间不足而一开始就要求一个大块头的array了。

C++STL序列式容器——vector容器详解_第2张图片

①vector的实现技术:
vector的实现技术,关键在于其对大小的控制以及重新配置时的数据移动效率,一旦vector旧空间满了,如果客户每新增一个元素,vector内部只是扩充一个元素的空间,实为不智,因为只要是扩充空间(不论多大),其步骤就是“配置新空间-数据移动-释放旧空间”的大工程,时间成本很高,所以为了降低空间配置时的速度成本,vector实际配置的大小可能比客户端需求大一些,以备将来可能的扩充,这边是容量的概念。换句话说,一个vector的容量永远大于或等于其大小,一旦容量等于大小,便是满载,下次再有新增元素,整个vector容器就得另觅居所(配置新空间-数据移动-释放旧空间)。
②vector迭代器:
vector维护一个线性空间,所以不论元素的型别如何,普通指针都可以作为vector的迭代器,因为vector迭代器所需要的操作行为,如operaroe*, operator->, operator++, operator–, operator+, operator-, operator+=, operator-=, 普通指针天生具备。vector支持随机存取,而普通指针正有着这样的能力。所以vector提供的是随机访问迭代器(Random Access Iterators).
③vector的数据结构:
vector所采用的数据结构非常简单,线性连续空间,它以两个迭代器_Myfirst和_Mylast分别指向配置得来的连续空间中目前已被使用的范围,并以迭代器_Myend指向整块连续内存空间的尾端。
④注意:

所谓动态增加大小,并不是在原空间之后续接新空间(因为无法保证原空间之后尚有可配置的空间),而是分配一块更大的内存空间,然后将原数据拷贝新空间,并释放原空间。因此,对vector的任何操作,一旦引起空间的重新配置,指向原vector的所有迭代器就都失效了。这是程序员容易犯的一个错误,务必小心。

二.vector常用操作

①vector构造函数

注:使用vector容器时,需包含头文件#include < vector>

函数 解释
vector< T > v; 采用模板实现类实现(显示实例化),默认构造函数,
vector(v.begin(), v.end()); 将v[begin(), end())区间中的元素拷贝给本身
vector(n, elem); 构造函数将n个elem拷贝给本身。
vector(const vector &vec); 拷贝构造函数,拿另一个vector对象初始化本身

实例:vector构造函数

#include  
#include 
using namespace std;
void printvector(const vector<int>& vec)
{
    for (vector<int>::const_iterator it = vec.begin(); it != vec.end(); ++it)
    {
        cout << *it;
    }
    cout << endl;
}
int main()
{
    vector<int> v1 = { 1,2,3 };//采用模板实现类实现(显示实例化),默认构造函数,
    vector<int> v2(6,1);//构造函数将n个elem拷贝给本身。
    int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
    vector<int> v3(arr, arr+sizeof(arr)/ sizeof(arr[0]));//将v[begin(), end())区间中的元素拷贝给本身
    vector<int> v4(v1);//拷贝构造函数,拿另一个vector对象初始化本身
    printvector(v1);
    printvector(v2);
    printvector(v3);
    printvector(v4);
    return 0;
}

C++STL序列式容器——vector容器详解_第3张图片

②特性操作

函数原型 解释
size_t max_size() const; 返回容器的最大长度
size_t capacity() const; 返回容器的容量。
size_t size() const; 返回容器的实际大小(已使用的空间)。
bool empty() const; 判断容器是否为空。
void clear(); 清空容器。
void reserve(size_t size); 容器预留size个元素长度,预留位置不初始化,元素不可访问,作用:预开辟空间,避免多次重新分配空间
void shrink_to_fit(); 将容器的容量降到实际大小(需要重新分配内存)。
void resize(size_t size); 把容器的实际大小置为size,如果size<实际大小,会截断多出的部分;如果size>实际大小,则以默认值0填充新位置
void resize(size_t size,const T &value); 把容器的实际大小置为size,如果size<实际大小,会截断多出的部分;如果size>实际大小,就用value填充。

实例:reserve和resize辨析:

#include  
#include 
using namespace std;
void printVector(const vector<int>& vec)//形参使用const,避免被修改
{   //const_iterator只读迭代器
	for (vector<int>::const_iterator it = vec.begin(); it != vec.end(); ++it)
	{
		cout << *it;
	}
	cout << endl;
}
void test()
{
    //1.resize开辟空间,并初始化
	//2.reserve开辟空间,但不初始化

	vector<int> v2;
	v2.push_back(1);//尾部插入一个元素
	v2.push_back(2);
	v2.push_back(3);

	cout << "size:" << v2.size() << endl;
	v2.resize(5);//重新指定容器的长度为num,若容器变长,则以默认值填充新位置。
	cout << "size:" << v2.size() << endl;
	printvector(v2);
	v2.resize(2);//如果容器变短,则末尾超出容器长度的元素被删除。
	cout << "size:" << v2.size() << endl;
	printvector(v2);
	v2.resize(5,6);//如果容器变长,也可以用value填充
	cout << "size:" << v2.size() << endl;
	printvector(v2);

	v2.reserve(20);
	v2.push_back(20);
	printvector(v2);//容器变长了,但并没有初始化
	cout << "size:" << v2.size() << endl;
	cout << "capacity:" << v2.capacity() << endl;

	//cout << v2[10] << endl;//err,不能访问未初始化的空间
}
int main()
{
	test();
	return 0;
}

C++STL序列式容器——vector容器详解_第4张图片

reserve作用:预开辟空间,避免多次重新分配空间
如果不加reserve预开辟空间,我们看看会重新分配多少次空间。

#include  
#include 
using namespace std;
void test()
{
    vector<int> v;
	int* p = NULL;
	int num = 0;

	for (int i = 0; i < 10001000; i++)
	{
		v.push_back(i);
		if (p != &v[0])
		{
			p = &v[0];
			num++;
		}
	}
	cout << "num=" << num << endl;
	return 0;
}
int main()
{
	test();
	return 0;
}

在这里插入图片描述
加上v.reserve(10001000);
在这里插入图片描述

③元素操作

函数原型 解释
T &operator[](size_t n); 通过[]访问元素,如果越界,不抛异常,程序直接挂掉
T &at(size_t n); 通过at方法获取下标为n的元素,如果n越界,抛出out_of_range异常
T *data(); 返回容器中动态数组的首地址。
const T *data() const; 返回容器中动态数组的首地址。
T &front(); 返回第一个元素。
T &back(); 返回,最后一个元素。

④赋值操作

作用:通过重载赋值运算符operator=和成员函数assign(),给已存在的容器赋值,将覆盖容器中原有的内容。

函数原型 解释
vector &operator=(const vector &v); 把容器v赋值给当前容器。
assign(beg, end); 将[beg, end)区间中的数据拷贝赋值给本身。
void assign(const size_t n, const T& value); ; 将n个value拷贝赋值给本身。
void assign(const size_t n, const T& value); 把n个value给容器赋值。

⑤交换操作

表格一览:

函数原型 解释
void swap(vector &v); 把当前容器与v交换。交换的是动态数组的地址。作品:收缩内存空间。

swap妙用:收缩内存空间

#include  
#include 
using namespace std;
void test()
{
    vector<int> v;
	for (int i = 0; i < 100000; i++)
	{
		v.push_back(i);
	}

	cout << "v的容量:" << v.capacity() << endl;
	cout << "v的实际大小:" << v.size() << endl;
	cout << "----------------" << endl;

	v.resize(10);//将容器的实际大小设置为10,但容量没变
	cout << "resize后v的容量:" << v.capacity() << endl;
	cout << "resize后v的实际大小:" << v.size() << endl;
	cout << "----------------" << endl;

	vector<int>(v).swap(v);//匿名对象和v对象交换
	//匿名对象调用拷贝构造,匿名对象的容量和实际大小均为10,此时在与v进行交换,做到了收缩内存的目的
	cout << "与匿名对象交换后v的容量:" << v.capacity() << endl;
	cout << "与匿名对象交换后v的实际大小:" << v.size() << endl;
}
int main()
{
	test();
	return 0;
}

C++STL序列式容器——vector容器详解_第5张图片

⑥比较操作

表格一览:

函数原型 解释
bool operator == (const vector< T > & v) const;
bool operator != (const vector< T > & v) const;

⑦插入和删除操作

表格一览:

函数原型 解释
iterator insert(iterator pos, const T& ele); 在指定位置插入一个元素ele 返回指向插入元素的迭代器。
iterator insert(const_iterator pos, int count,ele); 迭代器指向位置pos插入count个元素ele.返回指向插入元素的迭代器。
void push_back(const T& ele); 尾部插入元素ele
void pop_back(); 删除最后一个元素
iterator erase(const_iterator start, const_iterator end); 删除迭代器从start到end之间的元素,返回下一个有效的迭代器。
iterator erase(const_iterator pos); 删除迭代器指向的元素,返回下一个有效的迭代器。

实例:vector插入和删除操作

#include  
#include 
using namespace std;
void printVector(const vector<int>& vec)//形参使用const,避免被修改
{   //const_iterator只读迭代器
	for (vector<int>::const_iterator it = vec.begin(); it != vec.end(); ++it)
	{
		cout << *it;
	}
	cout << endl;
}
void test()
{
    vector<int> v;
	for (int i = 0; i < 5; i++)
	{
		v.push_back(i + 1);//尾部插入元素
	}

	printVector(v);//1 2 3 4 5

	v.insert(v.begin() + 1, 2, 100);//在第二个元素前插入2个100
	printVector(v);//1 100 100 2 3 4 5

	v.pop_back();//尾部删除一个元素
	printVector(v);//1 100 100 2 3 4
	cout << "-------------" << endl;
	v.erase(v.begin());//删除第一个元素
	printVector(v);//100 100 2 3 4

	vector<int>::const_iterator it=v.erase(v.begin() + 1, v.end() - 1);//删除从第二个元素到倒数第二个元素,返回下一个有效迭代器
	printVector(v);//100 4

	v.insert(it, 66);
	printVector(v);//100 66 4
}
int main()
{
    test();
	return 0;
}

C++STL序列式容器——vector容器详解_第6张图片

你可能感兴趣的:(C++核心编程,c++,开发语言,学习,笔记)