STL-09-forwardlist

本文详细探讨了C++11中的forward_list容器,它以单链表实现,强调空间效率和插入操作的高效性。文章比较了与list的差异,解释了forward_list在特定场景下的优势,以及如何利用其成员函数进行操作和性能测试实例。
摘要由CSDN通过智能技术生成

    forward_list 是 C++ 11 新添加的一类容器,其底层实现和 list 容器一样,采用的也是链表结构,只不过 forward_list 使用的是单链表,而 list 使用的是双向链表(如图 1 所示)。
在这里插入图片描述
    通过图 1 不难看出,使用链表存储数据最大的特点在于,其并不会将数据进行集中存储(向数组那样),换句话说,链表中数据的存储位置是分散的、随机的,整个链表中数据的线性关系通过指针来维持。

    因此,forward_list 容器具有和 list 容器相同的特性,即擅长在序列的任何位置进行插入元素或删除元素的操作,但对于访问存储的元素,没有其它容器(如 array、vector)的效率高。

    另外,由于单链表没有双向链表那样灵活,因此相比 list 容器,forward_list 容器的功能受到了很多限制。比如,由于单链表只能从前向后遍历,而不支持反向遍历,因此 forward_list 容器只提供前向迭代器,而不是双向迭代器。这意味着,forward_list 容器不具有 rbegin()、rend() 之类的成员函数。

    那么,既然 forward_list 容器具有和 list 容器相同的特性,list 容器还可以提供更多的功能函数,forward_list 容器有什么存在的必要呢?

    当然有,forward_list 容器底层使用单链表,也不是一无是处。比如,存储相同个数的同类型元素,单链表耗用的内存空间更少,空间利用率更高,并且对于实现某些操作单链表的执行效率也更高。

效率高是选用 forward_list 而弃用 list 容器最主要的原因,换句话说,只要是 list 容器和 forward_list 容器都能实现的操作,应优先选择 forward_list 容器。

接下来验证一下效率的问题

void main()
{
	cout << "list插入时间:";
	list<int> m_list;
	clock_t t1 = clock();
	for (int i = 1; i < 100000; i++)
	{
		m_list.push_back(i);
	}
	clock_t t2 = clock();
	cout << t2 - t1 << endl;

	

	cout << "\nforwardlist插入时间:";
	forward_list<int> m_forwardlist;
	clock_t t3 = clock();
	for (int i = 1; i < 100000; i++)
	{
		m_forwardlist.push_front(i);
	}
	clock_t t4 = clock();
	cout << t4 - t3 << endl;


	cout << endl;
	clock_t t5 = clock();
	for (auto node : m_list)
	{
	}
	clock_t t6 = clock();

	cout << "遍历list时间:" << t6 - t5 << endl;


	clock_t t7 = clock();
	for (auto node : m_forwardlist)
	{
	}
	clock_t t8 = clock();

	cout << "遍历forwardlist时间:" << t8 - t7 << endl;
	
	system("pause");
}

结果:
在这里插入图片描述
从结果可以看出forwardlist完胜list.

forward_list容器的创建

    由于 forward_list 容器以模板类 forward_list(T 为存储元素的类型)的形式被包含在<forward_list>头文件中,并定义在 std 命名空间中。因此,在使用该容器之前,代码中需包含下面两行代码:

#include <forward_list>
using namespace std;

std 命名空间也可以在使用 forward_list 容器时额外注明,两种方式都可以。

创建 forward_list 容器的方式,大致分为以下 5 种。

  1. 创建一个没有任何元素的空 forward_list 容器:
std::forward_list<int> values;

由于 forward_list 容器在创建后也可以添加元素,因此这种创建方式很常见。

  1. 创建一个包含 n 个元素的 forward_list 容器:
std::forward_list<int> values(10);

通过此方式创建 values 容器,其中包含 10 个元素,每个元素的值都为相应类型的默认值(int类型的默认值为 0)。

  1. 创建一个包含 n 个元素的 forward_list 容器,并为每个元素指定初始值。例如:
std::forward_list<int> values(10, 5);

如此就创建了一个包含 10 个元素并且值都为 5 个 values 容器。

  1. 在已有 forward_list 容器的情况下,通过拷贝该容器可以创建新的 forward_list 容器。例如:
std::forward_list<int> value1(10);
std::forward_list<int> value2(value1);

注意,采用此方式,必须保证新旧容器存储的元素类型一致。

  1. 通过拷贝其他类型容器(或者普通数组)中指定区域内的元素,可以创建新的 forward_list 容器。例如:
//拷贝普通数组,创建forward_list容器
int a[] = { 1,2,3,4,5 };
std::forward_list<int> values(a, a+5);
//拷贝其它类型的容器,创建forward_list容器
std::array<int, 5>arr{ 11,12,13,14,15 };
std::forward_list<int>values(arr.begin()+2, arr.end());//拷贝arr容器中的{13,14,15}
成员函数功能
before_begin()返回一个前向迭代器,其指向容器中第一个元素之前的位置。
begin()返回一个前向迭代器,其指向容器中第一个元素的位置。
end()返回一个前向迭代器,其指向容器中最后一个元素之后的位置。
cbefore_begin()和 before_begin() 功能相同,只不过在其基础上,增加了 const 属性,不能用于修改元素。
cbegin()和 begin() 功能相同,只不过在其基础上,增加了 const 属性,不能用于修改元素。
cend()和 end() 功能相同,只不过在其基础上,增加了 const 属性,不能用于修改元素。
empty()判断容器中是否有元素,若无元素,则返回 true;反之,返回 false。
max_size()返回容器所能包含元素个数的最大值。这通常是一个很大的值,一般是 232-1,所以我们很少会用到这个函数。
front()返回第一个元素的引用。
assign()用新元素替换容器中原有内容。
push_front()在容器头部插入一个元素。
emplace_front()在容器头部生成一个元素。该函数和 push_front() 的功能相同,但效率更高。
pop_front()删除容器头部的一个元素。
emplace_after()在指定位置之后插入一个新元素,并返回一个指向新元素的迭代器。和 insert_after() 的功能相同,但效率更高。
insert_after()在指定位置之后插入一个新元素,并返回一个指向新元素的迭代器。
erase_after()删除容器中某个指定位置或区域内的所有元素。
swap()交换两个容器中的元素,必须保证这两个容器中存储的元素类型是相同的。
resize()调整容器的大小。
clear()删除容器存储的所有元素。
splice_after()将某个 forward_list 容器中指定位置或区域内的元素插入到另一个容器的指定位置之后。
remove(val)删除容器中所有等于 val 的元素。
remove_if()删除容器中满足条件的元素。
unique()删除容器中相邻的重复元素,只保留一个。
merge()合并两个事先已排好序的 forward_list 容器,并且合并之后的 forward_list 容器依然是有序的。
sort()通过更改容器中元素的位置,将它们进行排序。
reverse()反转容器中元素的顺序。

看部分使用方法:

void main()
{

	forward_list<int> m_forwardlist{1,2,3};
	for (auto node :m_forwardlist)
	{
		cout << node << " ";
	}
	cout << endl;

	m_forwardlist.push_front(4);//插入到链表的最前面
	for (auto node : m_forwardlist)
	{
		cout << node << " ";
	}
	cout << endl;

	m_forwardlist.insert_after(m_forwardlist.before_begin(), 5);
	for (auto node : m_forwardlist)
	{
		cout << node << " ";
	}
	cout << endl;

	cout << "m_forwardlist的大小:" << std::distance(m_forwardlist.begin(), m_forwardlist.end()) << endl;
	system("pause");
}

结果;
在这里插入图片描述
并且,forward_list 容器迭代器的移动除了使用 ++ 运算符单步移动,还能使用 advance() 函数,比如:
在这里插入图片描述
在这里插入图片描述

更多的用法可以参考:C++基础之forward_list之补充

最后源码剖析可以参考list的源码。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

发如雪-ty

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值