STL — list容器用法的详解

 list容器用法的详解 

                                                                      




上一个博客我们学习了vector的使用我们也了解到了vector的优缺点,它的优点就支持随机访问,查找效率极高. 但是我

们也知道 vector插入的效率其实并不是特别的高,那么今天我们来瞧瞧list容器. 我们最好是找到他们之间的区别,比较

他们的应用场景. 相对于vector容器的连续线性空间,list是一个双向链表,它有一个重要性质:插入操作和删除操作都

不会造成原有的list迭代器 效,每次插入或删除一个元素就配置或释放一个元素空间。也就是说,对于任何位置的元素

插入或删除,list永远是常数时间 接下来更重要的事情使我们要了解到list是如何使用的..  打起精神!


list容器初始化

                                                                        

容器初始化的方式:


void Test()
{
	list<int> arr;
	//创建一个空的list

	list<int> arr2(arr);
	//通俗易懂的讲法就是拷贝构造.

	list<int> arr3(5);
	//创建5个元素的List,每个元素的值由默认构造函数构成.

	list<int> arr4(5, 1); 
	//创建5个元素的list,每个元素的值由你指定.

	list<int> arr(arr4.begin(),arr4.end());
	//由迭代器创建list,迭代区间为[begin,end];

	system("pause");
}


list容器的大小
                                                                          
       
判断list容器大小的函数如下:

Int size() const:                    返回容器元素个数


bool empty() const:                  判断容器是否为空,若为空则返回true


size_type max_size() const noexcept; 返回该容器可以容纳元素个数的大小.



代码演示:

void Test()
{

	list<int> arr(5, 1);

	cout << "该list容器是否为空:" << arr.empty() << endl;

	//注意empty()在容器为空,如果为空返回1

	int i = arr.size();

	//size()函数返回List当中的元素个数.

	cout << "该list容器内的元素个数为: " << i << endl;

	int count = arr.max_size();

	//max_size()返回list中最大容纳的个数.

	cout << "该list容器可最大容纳元素个数为: " << count << endl;

}

运行结果:



                                                      

list容器的遍历

                                                                             

用于list容器遍历的函数:


iterator begin():返回首元素的迭代器指针

iterator end():返回尾元素之后位置的迭代器指针

reverse_iterator rbegin():返回尾元素的逆向迭代器指针,用于逆向遍历容器

reverse_iterator rend():返回首元素前一个位置的迭代器指针

reference front():返回首元素的引用

reference back():返回尾元素的引用 


代码演示:


1 begin  end front back


void Test()
{
	list<int> arr(5, 1);

	list<int>::iterator it;

	//利用iterator遍历List容器

	cout << "原始时的list的元素分别为:";

	for (it = arr.begin(); it != arr.end(); it++)
	{
		cout << *it << " ";
	}

	cout << endl;

	int& i = arr.front();

	//front()返回首元素的引用

	int& j = arr.back();

	//back()返回尾元素的引用

	i = 100;

	j = 200;

	cout << "改变后的list的元素分别为:";

	for (it = arr.begin(); it != arr.end(); it++)
	{
		cout << *it << " ";
	}

	cout << endl;

}


运行结果:




2 rbegin rend


void Test()
{

	list<int> arr;

	for (int i = 1; i <= 5; ++i)
	{
		arr.push_back(i);
	}

	cout << "该List容器的元素分别为: ";

	for (list<int>::reverse_iterator it = arr.rbegin(); it != arr.rend(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

}

运行结果:



还有几个C++11里面的cbegin(),cend(),crbegin(),crend(),其实他们就是给上面说过的begin(),end(),rbegin(),


rend()返回值加上const属性. 大家思考一下灵活运用.


list容器的插入

                                                                            

list容器插入函数如下:


void push_back(const T& x):list元素尾部增加一个元素x

void push_front(const T& x):list元素首元素前添加一个元素X

iterator insert(iterator it, const T& x ):在迭代器指针it前插入元素x,返回x迭代器指针

void insert(iterator it,size_type n,const T& x):迭代器指针it前插入n个相同元素x

void insert(iterator it,const_iterator first,const_iteratorlast):把[first,last)间的元素插入迭代器指针it


代码演示:


1 push_back push_front


	list<int> arr;

	//利用push_back进行后插入元素.

	arr.push_back(1);

	arr.push_back(2);

	arr.push_back(3);

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

	//利用push_front进行前插入元素.

	arr.push_front(10);

	arr.push_front(200);

	arr.push_front(3000);

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

}
运行结果:



2 insert


void Test()
{


	list<int> arr(5, 1);

	list<int>::iterator it;

	it = arr.begin();

	arr.insert(it, 100);

	//对iterator指向的位置前面插入元素

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

	arr.insert(++it, 3, 200);

	//对iterator指向的位置前面出入n个相同的元素.

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

}


运行结果:



3  insert

void Test()
{
	list<int> arr(5, 1);

	list<int> arr2(5, 2);

	list<int>::iterator it;

	it = arr.begin();

	arr.insert(++it, arr2.begin(), arr2.end());

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;
}



运行结果:




list容器的删除

                                                                                 

list容器删除的函数;


void pop_back():删除容器尾元素,当且仅当容器不为空

void pop_front():删除容器首元素,当且仅当容器不为空

void remove(const T& x):删除容器中所有元素值等于x的元素

void clear():删除容器中的所有元素

iterator erase(iterator it):删除迭代器指针it对应的元素

iterator erase(iterator first,iterator last):删除迭代器指针[first,last)间的元素

代码演示:

1 pop_back pop_front

void Test()
{
	list<int> arr;

	arr.push_back(1);

	arr.push_back(2);

	arr.push_back(3);

	arr.push_back(4);

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

	//头删 pop_front() 和 尾删pop_back()的用法

	arr.pop_front();

	cout << "经过头删后该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

	arr.pop_back();

	cout << "经过尾删后该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;
}


运行结果:





2 remove clear

void Test()
{
	list<int> arr;

	arr.push_back(1);

	arr.push_back(2);

	arr.push_back(3);

	arr.push_back(2);

	arr.push_back(4);

	arr.push_back(2);


	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

	//使用remove删除掉所以等于x的元素.

	arr.remove(2);

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

	//使用clear()函数删除掉容器内所有元素.

	arr.clear();

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;
}

运行结果:




3  erase


void Test()
{


	list<int> arr;

	arr.push_back(1);

	arr.push_back(2);

	arr.push_back(3);

	arr.push_back(4);

	arr.push_back(5);

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	cout << endl;

	//使用erase删除指定的位置

	arr.erase(arr.begin());

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	arr.erase(arr.begin(), arr.end());

	cout << endl;

	arr.erase(arr.begin(), arr.end());

	cout << "该List容器的元素分别为: ";

	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}

	arr.erase(arr.begin(), arr.end());

	cout << endl;

}

运行结果:






list容器的操作

                                                                           


容器操作函数如下:

c1.swap(c2);                    将c1和c2交换。

c1.merge(c2)                    合并2个有序的链表并使之有序,从新放到c1里,释放c2。

c1.merge(c2,comp)               合并2个有序的链表并使之按照自定义规则排序之后从新放到c1中,释放c2。

c1.splice(c1.beg,c2,c2.beg)     将c2的beg位置的元素连接到c1的beg位置,并且在c2中施放掉beg位置的元素

unique()                        删除相邻的元素

c.sort()                        将链表排序,默认升序


代码演示:

1.swap  merge unique 

void print(list<T>& arr)
{
	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}
	cout << endl;
}

void Test()
{


	list<int> arr;

	list<int> arr2(5, 1);

	arr.push_back(1);

	arr.push_back(2);

	arr.push_back(3);

	arr.push_back(4);

	arr.push_back(5);

	cout << "arr List容器的元素分别为: ";
	print(arr);

	cout << "arr2 List容器的元素分别为: ";
	print(arr2);

	arr.swap(arr2); //交换两个链表.

	cout << endl;
	cout << "调用swap函数之后." << endl;
	cout << endl;

	cout << "arr List容器的元素分别为: ";
	print(arr);

	cout << "arr2 List容器的元素分别为: ";
	print(arr2);

	arr.merge(arr2);//合并两个链表并让它有序.

	cout << endl;
	cout << "调用merge函数之后." << endl;
	cout << endl;

	cout << "arr List容器的元素分别为: ";
	print(arr);

	cout << "arr2 List容器的元素分别为: ";
	print(arr2);

	arr.unique(); //删除所有连续并且相同的1

	cout << endl;
	cout << "调用unique函数之后." << endl;
	cout << endl;

	cout << "arr List容器的元素分别为: ";
	print(arr);
}

运行结果:




2.splice sort assign

template<class T>
void print(list<T>& arr)
{
	for (list<int>::iterator it = arr.begin(); it != arr.end(); ++it)
	{
		cout << *it << " ";
	}
	cout << endl;
}

void Test()
{
	list<int> arr;

	list<int> arr2(5, 1);

	arr.push_back(1);

	arr.push_back(2);

	arr.push_back(3);

	arr.push_back(4);

	arr.push_back(5);

	cout << "arr List容器的元素分别为: ";
	print(arr);

	cout << "arr2List容器的元素分别为: ";
	print(arr2);

	arr2.assign(5, 2); //将arr2中的元素重新初始化,使用方式跟构造相似.

	cout << endl;
	cout << "调用assign函数之后." << endl;
	cout << endl;

	cout << "arr2List容器的元素分别为: ";
	print(arr2);

	arr.splice(arr.begin(), arr2, arr2.begin(), arr2.end());
	//将arr2的所有元素,连接到arr.begin()前面,并释放掉arr2所有元素

	cout << endl;
	cout << "调用splice函数之后." << endl;
	cout << endl;

	cout << "arr List容器的元素分别为: ";
	print(arr);

	cout << "arr2List容器的元素分别为: ";
	print(arr2);

	arr.sort(); //排序就不说了.

	cout << endl;
	cout << "排序后 arr List容器的元素分别为: ";
	print(arr);
}

运行结果:



总结


我们发现比较一下list和vector之后就可以看到,list中提供了push_front,pop_front. 为什么vector没有? 其实稍微

想一下就能 明白,肯定是效率问题,vector插入的时候,后面的元素都要移动,而list不用. 所以插入效率list远高于

vector. 但是在查找和 遍历vector强于list.目前就学习到了这两个容器,剩下的map,set... 等后面我熟练get以后,我

会继续写出来的.


其实当你写一遍博客后,对这些所有的用法也差不多都明白了,以后在使用的过程中也会逐渐的得心应手,所以呢,我

们在学会一 个东西,我觉得最重要的东西就是积累,以后就算是完就赶紧回来看一下,虽然东西很简单可能都不用写博

客,我们应该就不要怕 麻烦,好好的积累.







  • 4
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: list、vector、map、set 是标准模板库(STL)中的容器,分别用于不同的数据存储和访问方式。 1. list(链表)是一个双向链表,可以根据需要在任意位置插入和删除元素。它没有固定大小,可以根据需要动态分配内存。由于是链表结构,所以访问元素的效率较低,但在插入和删除操作上非常高效。 2. vector(动态数组)是一个支持随机访问的连续内存的数组,可以在尾部追加元素,并且能够根据需要动态分配内存。由于是连续内存,所以在访问元素时效率较高,但在插入和删除操作上相对较慢。 3. map(映射)是一个存储键值对的关联容器,通过键(key)来访问值(value)。它内部实现了一棵红黑树(平衡二叉树),保证了以对数时间复杂度执行查找、插入和删除操作。键是唯一的,可以用于对数据进行有序存储和遍历。 4. set(集合)是一个存储唯一元素的容器,内部实现也是基于红黑树。它自动对元素进行排序,并且保证元素的唯一性。可以利用set进行集合运算,如并集、交集、差集等。 这四种容器在不同的场景下有不同的使用方式: - 如果需要经常进行插入和删除操作,并且不关心元素的顺序,则可以选择使用list。 - 如果需要频繁访问元素,不需要进行插入和删除操作,或者只在尾部进行插入和删除,则可以选择使用vector。 - 如果需要按照键来查找和存储数据,则可以选择使用map。 - 如果需要存储唯一元素,并且需要对元素进行排序和集合运算,则可以选择使用set。 当然,具体的选择还会根据实际需求和性能要求进行权衡。 ### 回答2: list、vector、map、set都是C++标准模板库(STL)中的容器,有着不同的特点和用法。 1. list(链表)是双向链表的实现,其中的元素可以随意插入和删除。它没有随机访问的功能,只能通过迭代器进行跳跃式访问。它适用于需要频繁的插入和删除操作,并且不需要随机访问的场景。 2. vector(向量)是动态数组的实现,支持随机访问。在vector中进行插入和删除操作需要移动其他元素,所以效率可能会低于list。但是它具有随机访问的能力,并且在末尾插入和删除元素的效率较高。vector适用于需要频繁的随机访问和在末尾插入删除元素的场景。 3. map(映射)是一种由键和值对组成的容器,通过键进行查找和插入操作。map内部通过红黑树实现,所以键值对是按照键的顺序来存储的。map中的每个键是唯一的,如果插入一个已存在的键,将会覆盖原有的值。map适用于需要按键进行查找的场景。 4. set(集合)是一种由唯一元素组成的容器,元素按照一定的顺序进行存储。set内部也是通过红黑树实现,所以元素是按照一定顺序排列的。set中的元素是唯一的,插入重复的元素将会被忽略。set适用于需要维护一组唯一元素并进行一些集合操作(如并集、交集等)的场景。 总结而言,list适用于频繁的插入和删除操作,vector适用于频繁的随机访问和末尾插入删除操作,而map和set适用于按键进行查找或维护一组唯一元素的场景。根据实际需求选择适当的容器可以提高代码的效率。 ### 回答3: list、vector、map和set是C++标准库中常用的容器类,它们分别用于不同的存储和访问数据的需求。 1. list(链表): - 使用双向链表实现,支持快速的插入和删除操作; - 不支持随机访问,只能通过迭代器依次访问元素; - 元素的添加和删除不会导致迭代器失效; - 适用于需要频繁插入和删除元素的场景,但对于随机访问的需求较少。 2. vector(动态数组): - 使用动态数组实现,支持随机访问; - 在尾部插入或删除元素的时间复杂度为常数,其他位置的插入和删除操作需要移动部分元素; - 内存分配是连续的,支持快速的随机访问; - 在需要经常随机访问元素的场景下使用较多。 3. map(有序映射): - 使用红黑树实现,基于键值对的有序映射; - 插入和查找操作的时间复杂度为对数时间复杂度; - map中的元素按照键值的顺序进行排列; - 适用于需要进行查找操作的场景,如字典、索引等。 4. set(有序集合): - 使用红黑树实现,表示一个有序的不重复元素集合; - 插入和查找操作的时间复杂度为对数时间复杂度; - set中的元素按照从小到大的顺序排列; - 适用于需要维护有序不重复元素的场景。 总结: - list适用于频繁插入和删除元素的场景; - vector适用于需要经常随机访问元素的场景; - map适用于需要查找操作的场景,按照键值有序排列; - set适用于需要维护有序不重复元素的场景。 实际使用中,根据具体的需求选择合适的容器可以提高程序的效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值