今日目录:
1:能够说出链表的存储结构和特点
2:能够说出链表的几种分类及各自的存储结构
3:能说出链表和数组的差异
4:完成实战演练题目
5:完成综合案例
1、概念及存储结构
问题:思考一下动态数组ArrayList存在哪些弊端?
1:插入,删除时间复杂度高
2:可能会造成内存空间的大量浪费
3:需要一块连续的存储空间,对内存的要求比较高,比如我们要申请一个1000M的数组,如果内存中没有连续的足够大的存储空间则会申请失败,即便内存的剩余可用空间大于1000M,仍然会申请失败。
结论:能否做到用多少内存空间就申请多少内存?
链表(Linked list)是一种物理存储单元上非连续、非顺序的存储结构,链表中的每一个元素称之为结点(Node),结点之间用指针(引用)连接起来,指针的指向顺序代表了结点的逻辑顺序,结点可以在运行时动态生成。每个结点包括两个部分:一个是存储数据元素的数据域,另一个是存储下一个结点地址的指针域。
链表能解决数组不能解决的事情吗?
1:链表天生就具备动态扩容的特点,不需要像动态数组那样先申请一个更大的空间,然后将原空间内的数据拷贝到新空间;能够避免内存空的大量浪费
2:链表不需要一块连续的内存空间,它通过指针将一组零散的内存块串联起来使用,所以如果我们申请一个1000M大小的链表,只要内存剩余的可用空间大于1000M,便不会出现问题。
但是需注意:存储同样的数据,链表要比数组耗费内存!
2、链表分类
链表根据其结点之间的连接形式我们又可分为:单链表,双向链表,循环链表,双向循环链表
2.1、单链表
单链表就是我们刚刚讲到的链表的最基本的结构,链表通过指针将一组零散的内存块串联在一起。。如图所示,我们把这个记录下个结点地址的指针叫作后继指针 next ,如果链表中的某个节点为p,p的下一个节点为q,我们可以表示为:p.next=q
链表中有两个结点是比较特殊的,它们分别是第一个结点和最后一个结点。我们习惯性地把第一个结点叫作头结点,把最后一个结点叫作尾结点。其中,头结点用来记录链表的基地址,有了它,我们就可以遍历得到整条链表。而尾结点特殊的地方是:指针不是指向下一个结点,而是指向一个空地址 NULL,表示这是链表上最后一个结点。
与数组一样,链表也支持数据的查找、插入和删除操作。
在进行数组的插入、删除操作时,为了保持内存数据的连续性,需要做大量的数据搬移,所以时间复杂度是 O(n)。而在链表中插入或者删除一个数据,我们并不需要为了保持内存的连续性而搬移结点,因为链表的存储空间本身就不是连续的。所以,在链表中插入和删除一个数据是非常快速的。
如图所示,针对链表的插入和删除操作,我们只需要考虑相邻结点的指针改变,所以插入删除的时间复杂度是 O(1)。
但是,有利就有弊。链表要想随机访问第 k 个元素,就没有数组那么高效了。因为链表中的数据并非连续存储的,所以无法像数组那样,根据首地址和下标,通过寻址公式就能直接计算出对应的内存地址,而是需要根据针一个结点一个结点地依次遍历,直到找到相应的结点,所以,链表随机访问的性能没有数组好,查询的时间复杂度是O(n)。
2.2、双向链表
单向链表只有一个方向,结点只有一个后继指针 next。而双向链表,顾名思义,它支持两个方向,每个结点不止有一个后继指针 next 指向后面的结点,还有一个前驱指针 prev 指向前面的结点,如图所示