细腻的链接:C++ list 之美的解读

细腻的链接:C++ list 之美的解读

前言:

小编在前几日刚写过关于vector容器的内容,现在小编list容器也学了一大部分了,小编先提前说一下学这部分的感悟,这个部分是我学C++以来第一次感到有难度的地方,特别是在list容器的模拟实现中便可以看出来,到时候再细说,总之就是我在这一部分感到有一点困难了,它很考察前面知识点,包括但不限于类和对象,模板,数据结构部分内容等等,这边话也不多说了,开启今日的学习时间。

文章目录

正文:

1.list的介绍

1.1.list是什么

list也和我们之前学过的某个数据结构息息相关,它就是:双链表,list的底层实际上就是带头双向循环链表,恰好小编这里有一个很形象的展示它的图片,这边我就分享出来了。

通过上图我们就可以看出list的具体结构,它就是我们之前学过的双向带头循环链表(双链表),此时的list和上次我讲述的vector是一样的,它们都是一个模版类,因为vector容器的类型可能都是不相同的,他可以接受int,float,double作为类型。甚至list同样也可以存储string容器,可谓是很全能,这便是模版带来的好处,可以帮助我们完成许多事情。

1.2.list功能一览

下面小编先给各位展示一下网站上对于list容器的简介,可能有很多读者朋友是第一次看我的文章,所以我还是放上这个网站的链接:list - C++ Reference,各位读者朋友可以在这个网站看到list容器更为详细的介绍,只不过很考验各位的英语功底,我还是那句话

下面小编就和之前STL容器讲解一样,开始对于list部分函数的介绍~

2.list功能讲述

2.1.list的构造

构造函数也是我们的老朋友,每一个容器都是有它自己的构造函数的,连构造函数都没有,这个容器创建起来也是没有一点意义,下面小编就分别讲述一下部分构造函数的使用,可能讲的不全,但是只要先把这个掌握了,那么在掌握其他就容器的多了。

构造函数(constructor) 接口说明
list (size_type n, const value_type& val =value_type()) 构造的list中包含n个值为val的元素
list() 构造空的list
list (const list& x) 拷贝构造函数
list (InputIterator fifirst, InputIterator last) 用[first, last)区间中的元素构造list
2.1.1.list(size_t n,const value_type& val = vlaue_type())

这个构造函数,乍一看感觉有点困难,其实它是很简单的,这个函数小编在之前的容器中也不止一次说过,它的功能其实就是构造并初始化n个val罢了,它的形参之所以看起来如此的复杂,其实就是因为它是一个模版类罢了,下面小编展示用法:

c++ 复制代码
std :: list<int> s1(10,1);  //这里我还是提醒一下,因为list是模板类,所以初始化必须显示初始化,告知编译器它内容的类型
2.1.2.list()

这个函数也是一个常客了,小编在前面也说过,它的功能其实就是构造并初始化一个空的list,下面我直接上代码:

c++ 复制代码
std :: list<int>();
2.1.3.list(const list& x)

这个函数也是比较好认识的,通过括号里面的内容,我们便可以知晓它是一个很典型的拷贝构造函数的写法,这个函数小编在类和对象(3)讲过,感兴趣的读者朋友可以去看一看(每日推销),废话不多说,小编展示它的用法:

c++ 复制代码
std :: list<int> s1;
std :: list<int> s2 = s1;  //也可以写成list<int>s2(s1),想写成啥就写成啥
2.1.4.list(Inputlterator first,Inputlterator last)

这个函数我也在之前讲过,看着这个有点复杂,其实作用是很简单的,它的作用无非就是用[first,last)区间(谨记这里是左闭右开区间)的因素构造list,下面我直接展示用法:

c++ 复制代码
std :: list<int> s1;
std :: list<int> s2(s1.begin(),s1.end());

以上便就是构造函数的用法,构造函数在list众多接口中苏是比较容易的,下面小编继续往后进行讲述。

2.2.迭代器(iterator)的使用

对于迭代器,想必读者朋友到这里就不陌生了,它是一个很像指针的东西,只不过它并非就是指针,这里我先预告一下,小编在list类的模拟实现中,将不会再把指针包装成迭代器了,具体情况请看我推出的模拟使用篇章(大概很久,因为string我还没写),下面小编就讲述一下关于迭代器函数的使用,突然想起来一个C++11新加入的内容我没讲,那就是范围for,因为范围for的底层是迭代器,这里既然有迭代器,自然可以支持范围for,所以各位读者朋友记住就好。

函数声明 接口声明
begin +end 返回第一个元素的迭代器 +返回最后一个元素下一个位置的迭代器
rebgin+rend 返回第一个元素的reverse_iterator,即end位置,返回最后一个元素下一个位置的reverse_iterator,即begin位置
re2.2.1.begin + end

这两个函数也属于我们的老朋友了,小编在之前的容器讲解中就多次讲述了这俩函数,这里它们的作用可能和前面不太相同,因为此时的list是链表结构,此时的begin指向的是第一个有意义的结点,而end不在是最后一个位置,而是指向的哨兵位,不然的话遍历会出现死循环的情况,并且会完整的遍历一遍,虽然此时的迭代器不在是指针,但它和指针的用法还是一直的,下面小编带领各位展示它的用法:

c++ 复制代码
using namespace std;
list<int> s1 = {1,2,3,4,5}; //这个操作小编没有在构造函数讲,实际上这个就是实现了一次隐式类型转换,小编在vector就说过,具体的可以看那篇文章。
auto it = s1.begin();
while(it != s1.end())
{
    cout << *it << " ";
    it++;
}
2.2.2.rbegin + rend

这两个函数也属于是老朋友了,它俩的作用就是和上面的函数截然相反,就是遍历的顺序相反了,这便是反向迭代器,至于反向迭代器的深层次讲解,小编将会在之后的栈和队列的容器中会详细讲述,下面直接展示用法:

c++ 复制代码
using namespace std;
list<int> s1 = {1,2,3,4,5}; 
auto it = s1.rbegin();
while(it != s1.rend())
{
    cout << *it << " ";
    it++;
}

以上便就是迭代器的相关内容,由于比较简单,我就一笔带过了,对于这部分内容我有个图片的形象展示,我便分享出来帮助各位更好的去理解。

2.3.list capacity

正如其名,这部分将会讲述关于list空间相关的函数,不过讲述的其实也不算很多了,因为关键的就那几个,下面小编直接开讲。

函数声明 接口说明
empty 检测list是否为空,是返回rue,否则返回false
size 返回ist中有效节点的个数
2.3.1.empty

这个函数自然不用多说了,它的名字就告知了各位它的用法,它的作用主要就是判断链表是否为空的,下面浅浅的展示用法:

c++ 复制代码
std :: list<int> s1 = {1,2,3};
if(s1.empty())
{
    cout << 1;  //这里不放图片了,结果显而易见,自然不会打印1
}
2.3.2.size

这个函数也是老朋友了,它的功能是检测数list里面有多少个有效结点,譬如上面的代码,就知道它的size应该是有3个,哨兵位不算有效结点,下面展示用法:

c++ 复制代码
std :: list<int> s1 = {1,2,3};
cout << s1.size() << endl; //这里自然打印三个,我就不放图片了

以上便就是小编对于空间函数的详解,可能有读者朋友好奇为啥子没有capcoaty,这里我只能说好好复习一下链表的知识,链表的特点就是随用随开辟,当我们创建新结点的时候就开辟一个新的,不用多开,所以这便是链表的便携之处,各位要知道,下面我们继续进入函数的讲解。

2.4.list element access

本来我不想讲述这个函数,但我看了看不讲吧可能很多读者朋友不懂,所以我还是讲讲吧,这个英文的翻译是链表的元素访问,所以它提供的函数自然就是访问链表中某些特定部分的函数,下面说说与它相关函数的讲解。

函数声明 接口说明
front 返回list的第一个节点中值的引用
back 返回list的最后一个节点中值的引用
2.4.1.front

通过这个函数的函数名不难看出,这个函数的功能应该是和第一个有意义的结点有关的,确实,这个函数的作用就是返回第一个有效结点的值,由于难度不大,小编直接讲述这个函数的

c++ 复制代码
std :: list<int> s1 = {1,2,3};
cout << s1.front() << endl;  //这个打印出是1,我就不截图了。
2.4.2.back

这个函数的作用其实也是很容易看出来的,这个单词的意思是有返回的意思,仔细想一想,到最后才可以返回,所以这个函数的作用就是返回最后一个结点的数据,下面还是展示用法:

c++ 复制代码
std :: list<int> s1 = {2,3,4};
cout << s1.back() << endl;

以上便就是关于元素访问函数的讲解,下面我们继续往后讲解。

2.5.list modifiers

下面就是今天小编要讲述的最后一个接口,它是关于list的增删查改函数的讲解,下面不多废话,直接进入这个接口的详解。

函数声明 接口说明
push_front 在list首元素前插入值为val的元素
pop_front 删除list中第一个元素
push_back 在list尾部插入值为val的元素
pop_back 删除list中最后一个元素
insert 在ist position位置中插入值为val的元素
erase 删除list position位置的元素
swap 交换两个ist中的元素
clear 清空ist中的有效元素
2.5.1.push_front

这个函数想必学过数据结构的朋友们都不陌生,它就是在前面容器都没有出现过的头插函数,旨在头部插入新结点,当然,肯定不是在哨兵位前面插入结点,它是在第一个有效元素的前面插入新的结点,小编在双链表的时候也讲述过这个函数的写法,同样的,我会在模拟实现的时候再次实现头插,各位读者朋友敬请期待,下面给上用法:

c++ 复制代码
std :: list<int> s1 = {1,2,3};
s1.push_front(1);  //这样的话会打印出1,1,2,3,我就不用图片展示了
2.5.2.pop_front

有插入,自然就有删除函数,这个函数的作用就是起到头删的作用,当然不是删哨兵位,而是删除第一个有效结点如果链表里面有结点的话,那么会把第一个有效的结点进行删除,倘若里面只有一个哨兵位节点的话,那么是会报错的,下面小编展示用法:

c++ 复制代码
std :: list<int> s1 = {1,2,3};
s1.pop_front();  //这样的话会打印出2,3,我就不用图片展示了
2.5.3.push_back

有头插函数,自然就有我们的老朋友------尾插函数,这个函数通过名字就可以知道,它是往最后一个位置插入新的结点,由于难度不达,小编直接展示用法:

c++ 复制代码
std :: list<int> s1 = {1,3,4,5};
s1.push_back(1);  //这样的话会打印出1,3,4,5,1,我就不展示打印图了
2.5.4.pop_front

有尾插函数,自然就有尾删函数,这个函数的做用就是删除尾部的数据,当然,如果这个链表仅仅只有哨兵位一个结点的话,那么会报错,由于难度不大,我就直接上用法了:

c++ 复制代码
std :: list<int> s1 = {1,2,3,4};
s1.pop_back();  //这样的话仅仅会删掉4这个结点的数据
2.5.5.insert

这个函数也是个老熟人了,它的作用是在任意位置上插入数据,所以它既可以实现头插尾插,也可以实现任意插入数据,在之后的尾插函数和头插函数的模拟实现中,在最后我们可以只用实现一个insert函数就好了,用insert套用实现它们,下面小编展示它的用法:

c++ 复制代码
std :: list<int> s1 = {1,2,3,4};
s1.insert(s1.begin(),2);  //这是实现了头插
2.5.6.erase

既然有指定位置插入函数,那么肯定有指定位置删除函数,这个函数的作用同样也可以实现头删尾删,下面小编展示用法:

c++ 复制代码
std :: list<int> s1 = {1,2,3,4};
s1.erase(s1.begin());  //这里我就简单实现了一下头插函数
2.5.7.swap

swap函数,想必这个肯定不陌生了,我们在C语言阶段就多次使用到swap函数,我记着在数据结构堆的实现中我还用到了这个函数,实现了堆排序,在C++阶段我在模版的时候就用它作为引子,我忘记当时我说没说,swap函数在库中就拥有,只不过库里的swap函数最好不要用,因为代价是很伤的,具体情况请看我以后写的string模拟实现,下面展示它的用法:

c++ 复制代码
using namespace std;
list<int> s1 = {1,2,3,4};
list<int> s2 = {2,5,7,12};
s1.swap(s2);
2.5.8.clear

这个函数我之前在string应该说过...吧,这个函数作用就是起清空数据的作用,倘若我们写了一个很多结点的list,想要一次性释放,不放试试这个函数,直接就把所有结点给释放了,下面小编直接展示他的用法:

c++ 复制代码
using namespace std;
list<int> s1 = {1,2,3,4,5,51,1,2,3,4,4};
s1.clear();  //使用它会直接被清空(里面的结点)。

以上便就是小编将要讲述的增删查改函数,这就是小编将要讲述的最后一个功能,其实学多了容器,各位读者朋友不难发现,容器的功能的使用其实都是很相似的,只要我们掌握了其中一个容器的写法,剩余容器的写法便可以会轻易的掌握,难的是底层逻辑,我将会在不久的将来诞生出list容器的模拟实现文章,希望不会太久。

3.总结

本文到这里也是创作结束了,感觉周末的我,还是比较贪玩的,这篇文章我花了两天才写完,确实是我懈怠了,这里提醒那些爱看小说的读者朋友,千万不要沉迷于小说,不然会耽误很多事,对于文章有任何错误的地方,可以在评论区指出,我会定期逛评论区来给大家解惑,那么各位大佬们,我们下一篇文章见啦!

相关推荐
诚丞成1 分钟前
计算世界之安生:C++继承的文水和智慧(上)
开发语言·c++
Smile灬凉城66613 分钟前
反序列化为啥可以利用加号绕过php正则匹配
开发语言·php
lsx20240624 分钟前
SQL MID()
开发语言
Dream_Snowar27 分钟前
速通Python 第四节——函数
开发语言·python·算法
西猫雷婶29 分钟前
python学opencv|读取图像(十四)BGR图像和HSV图像通道拆分
开发语言·python·opencv
鸿蒙自习室29 分钟前
鸿蒙UI开发——组件滤镜效果
开发语言·前端·javascript
言、雲37 分钟前
从tryLock()源码来出发,解析Redisson的重试机制和看门狗机制
java·开发语言·数据库
TT哇44 分钟前
【数据结构练习题】链表与LinkedList
java·数据结构·链表
东风吹柳1 小时前
观察者模式(sigslot in C++)
c++·观察者模式·信号槽·sigslot
A懿轩A1 小时前
C/C++ 数据结构与算法【栈和队列】 栈+队列详细解析【日常学习,考研必备】带图+详细代码
c语言·数据结构·c++·学习·考研·算法·栈和队列