小解List的使用【C++】

小解List的使用【C++】

  • 一. List
    • 1.1. 与vector的不同
    • 1.2 与vector的使用不同
      • 1.2.1 迭代器失效
      • 1.2.2. insert
      • 1.2.3 erase
      • 1.2.4 sort
      • 1.3. 其他接口
  • 补充迭代器
    • 容器与迭代器的关系
    • 迭代器的类型

一. List

学习了STL,也已经到了List的内容
因为List与string以及vector比起来还是有很大不同的,所以这里稍微除了一篇小博客主要来讲解一下
相对于vector以及string,List中比较特殊的部分。

1.1. 与vector的不同

我们再学习这个List之前:
大部分人应该都已经知道list的底层实际上是:双向带头循环链表
这个其实以前博主再博客中实现过。

这里也不深究

string和vector实际上本质是顺序表

这两个类型的最大区别就是:
vector在内存中相当于动态开辟的数组,地址是连续的

链表中的节点则不是连续的,在内存中表现的是断开的,不连续的形式

所以vector的区别和使用上的区别也都是围绕着这个展开的。

与string有较大的不一样,因为空间不连续

1.2 与vector的使用不同

这里主要挑出来几个常用的功能
来体验一下vector与list的区别。

1.2.1 迭代器失效

还记得在vector中的迭代器失效吗

就是当使用迭代器进行insert和erase后,
重新使用迭代器对象进行操作
因为迭代器指向的位置虽然没变,但是指向的值却已经发生了变化。
在vs编译器下,直接限制了用户对erase与insert后的指针对象的使用。

#include
#include
#include"vector.h"
int main()
{
	std::vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	std::vector<int>::iterator begin = v1.begin();
	v1.erase(begin);
	while (begin != v1.end())
	{
		std::cout << *begin;
		begin++;
	}
}

就是上面的情况。

但是我们把目光转向list,看看list中是否存在迭代器失效的结果。

1.2.2. insert

这里我们直接进行测试把

int main()
{
	vector<int> v1;
	vector<int>::iterator it = v1.begin();
	v1.insert(it, 1);
	v1.insert(it, 2);
}

这里先用vector进行测试
很明显是行不通的,因为it被连续调用了两次。

小解List的使用【C++】_第1张图片

这里就用list进行测试:
小解List的使用【C++】_第2张图片

小解List的使用【C++】_第3张图片

这里能发现并没有报错。

这里其实我们细想一下也能明白

list使用insert时候
list中的地址指向的值没有发生变化
因为他们本身就是不连续的,数据插入后,他们指针指向的位置依旧还是原来的值。

所以不会出现和vector一样的情况——移动后,前面的元素,会取代原来的值所以指针指向的值就发生了变化。

所以用insert不会发生迭代器失效的问题。

这里又要提出一个小问题:
因为vector的地址是连续的,在insert中可以直接用:

v1.insert(it+5,2);

表示在迭代器的+5的位置处进行插入

这里vector能这么用,是因为vector的迭代器能支持加减,因为vector的地址是连续的

但是list的话因为地址不一样,所以是不是可能就不行了。

这里我们来试一下
小解List的使用【C++】_第4张图片

这里就非常明显的报错了。

所以我们list想在迭代器后面进行插入就要自己让迭代器去++;

比如这里:
想要到迭代器后的第三个位置进行插入

std::list<int>::iterator it=s1.begin();
for(int i=0;i<3;i++)
{
	it++;
}

1.2.3 erase

其实erase这里不用实验。想想就能知道了

erase的作用是删除节点。
那节点都删除了,迭代器指向的指针位置肯定也消失了。
所以这里就不实验了,毫无疑问。

那问题应该是怎么解决:
其实这里和vector里的一样:
it=it.erase();

it会自带返回迭代器
返回的指向对象正是迭代器删除的对象的下一个指向对象。

1.2.4 sort

我们知道在算法中有一个qsort算法。

但是在list中,list自带了一个qsort接口,方便用户进行排序。

但是这里我们并不提倡用这个qsort

因为:
sort():
算法中的sort效率远高于list的sort效率
数据量差的越多,效率差距越大

因为算法中的sort用的是快速排序

而因为list中的地址不是连续的,所以并不能用快速排序,只能用归并排序进行实现

这里我们能来进行测试以下:

这里把测试函数塞进来:

void test2()
{
	int N = 1000000;//测试的数字的多少
	std::vector<int>v1;
	//添加随机值给vector
	for (int i = 0; i < N; i++)
	{
		auto s = rand();
		v1.push_back(s);

	}

	//给list添加随机值
	std::list<int> l1;
	for (int i = 0; i < N; i++)
	{
		auto s = rand();
		l1.push_back(s);

	}
	
	//排序list并记录时间
	int begin1 = clock();
	l1.sort();
	int end1 = clock();
	
	//排序vector并记录时间
	int begin2 = clock();
	sort(v1.begin(), v1.end());
	int end2 = clock();

	std::cout << "vector " << end2 - begin2 << std::endl;
	std::cout << "list " << end1 - begin1 << std::endl;
}

在这里插入图片描述
这里我们能看到list和vector的效率差距还是十分大的。

所以还是不推荐用list的sort的

那我们想要排序list中的数怎么办。

我们可以将list中的数字拷贝进vector中,然后再vector排序完了后,赋值给list

void test3()
{
	int N = 1000000;
	std::vector<int>v1;
	for (int i = 0; i < N; i++)
	{
		auto s = rand();
		v1.push_back(s);

	}


	std::list<int> l1;

	for (int i = 0; i < N; i++)
	{
		auto s = rand();
		l1.push_back(s);

	}


	int begin1 = clock();
	sort(v1.begin(), v1.end());
	int end1 = clock();



	int begin2 = clock();
	std::vector<int>v2;
	for (auto i : l1)
	{
		v2.push_back(i);
	}
	sort(v2.begin(), v2.end());
	size_t i = 0;
	//将vector赋值给list
	for (auto& z : l1)
	{
		z = v2[i++];
	}
	int end2 = clock();


	std::list<int> l2;
	for (int i = 0; i < N; i++)
	{
		auto s = rand();
		l2.push_back(s);
	}


	int begin3 = clock();
	l2.sort();
	int end3 = clock();



	std::cout << "vector " << end1 - begin1 << std::endl;
	std::cout << "list_good " << end2 - begin2 << std::endl;
	std::cout << "list " << end3 - begin3 << std::endl;

}

1.3. 其他接口

这里的其他接口就不进行演示了
这里贴个网址自己去使用即可:
List的其他接口

补充迭代器

这里因为vector与list有许多的不同。

所以他们的迭代器有很大的不同,这里就补充一下迭代器。

容器与迭代器的关系

我们都知道:

容器使用来存储数据的

算法是用来处理数据的。

而迭代器是用来链接容器和算法之间的桥梁。

有了迭代器,算法就可以通过访问迭代器改变容器中的数据。

所以对于迭代器来说要通过容器的性质来进行设计。
容器的不同会导致迭代器的类型也不同

迭代器的类型

小解List的使用【C++】_第5张图片

这里的容器种类不同,迭代器的种类也不同。

迭代器的种类不同,我们能看到他们支持的运算符种类也不同。

你可能感兴趣的:(list,c++,数据结构)