链表理论基础
链表是一种通过指针串联在一起的线性结构,每一个节点由两部分组成,一个是数据域一个是指针域(存放指向下一个节点的指针),最后一个节点的指针域指向null(空指针的意思)。
链表的入口节点称为链表的头结点也就是head。
链表的类型:单链表、双链表、循环链表
链表的存储方式:
数组是在内存中是连续分布的,但是链表在内存中可不是连续分布的。
链表是通过指针域的指针链接在内存中各个节点
链表的定义
// 定义链表节点方式(单链表)
struct ListNode {
int val; // 节点上存储的元素
ListNode *next; // 指向下一个节点的指针
ListNode(int x) : val(x), next(NULL) {} // 节点的构造函数
};
链表的操作
删除节点
添加节点
性能分析
203.移除链表元素
建议: 本题最关键是要理解 虚拟头结点的使用技巧,这个对链表题目很重要。
使用虚拟头结点的方法可以使添加/删除链表中每个节点(包括头结点)的操作时保持一致的规则
方法一:直接使用原来的链表来进行移除节点操作
方法二:设置一个虚拟头结点在进行移除节点操作
定义多一个指针类型赋值(ListNode* cur=dummyhead);删除节点步骤,定义临时指针(
ListNode* tmp=cur->next;
cur->next=cur->next->next;
delete tmp;
)
707.设计链表(还没做)
建议: 这是一道考察 链表综合操作的题目,不算容易,可以练一练 使用虚拟头结点
接口组成:获取链表第index个节点的数值、在链表的最前面插入一个节点、在链表的最后面插入一个节点、在链表第index个节点前面插入一个节点、删除链表的第index个节点
206.反转链表
建议:先看我的视频讲解,视频讲解中对反转链表需要注意的点讲的很清晰了,看完之后大家的疑惑基本都解决了。
双指针法:清晰明了的方法
注意:先赋值pre再赋值cur;反转链表关键步骤(
temp=cur->next;
cur->next=pre;
pre=cur;
cur=temp;
)
递归法(需重写):相较双指针法而言比较难懂,建议根据已写出的双指针方法写出代码
需确定递归函数内的形参