引言:以下内容均只是个人看法,并无严格经过核对,如有异议,欢迎交流,发现错误,欢迎您的及时指正,感谢!本文所示代码均为java代码
本文中使用的结点定义语句如下
public class ListNode {
public int val;
public ListNode next;
ListNode(int x) {
val = x;
//这个一般作用不大,写了会更加规范
next = null;
}
}
单链表的遍历一般从第一个结点依次遍历整个链表,所以第一个结点十分重要,在本文中头节点指的是第一个存放数据的结点,与其他相关文章中关于它的定义可能不同,无需深究,理解相关操作的逻辑原理即可,本质上无较大差别。
我们来讨论一下链表反转的问题,给定一个链表的头结点,让你输出将它反转后的链表的头结点
1.建立虚拟头结点辅助反转
也可以称为头插法,这种方法我们借助了虚拟节点ans来帮助确认pre也就是反转后链表的头结点,利用了cur指针来对待处理链表中的每一个节点在反转链表中进行插入操作。
代码如下
/**
*
* @param head 需要反转的链表的头结点
* @return
*/
//建立虚拟头结点辅助反转
public static ListNode reverseList1(ListNode head){
//ans是虚拟头结点,cur始终指向待处理链表的第一个节点
ListNode ans=new ListNode(0);
ListNode cur=head;
while(cur!=null){
//附设一个next指针用来记录cur.next,以免在插入操作后cur的next找不到
ListNode next=cur.next;
cur.next=ans.next;
ans.next=cur;
cur=next;
}
return ans.next;
}
2.直接操作链表实现反转
直接操作链表的反转借用了两个指针,pre相当于实时作为反转链表的头指针,cur实时指向下一个需要接入反转链表的
我们由图可以看出,这个过程实际上就是不断把cur加入到左边pre的链表,并修改pre指针的位置
代码如下:
/**
*
* @param head 需要反转的链表的头结点
* @return
*/
//直接操作链表实现反转
public static ListNode reverseList2(ListNode head){
//cur指向下一个待反转的节点 ,pre作为反转了的部分的当前头结点
ListNode cur=head,pre=null;
while(cur!=null){
ListNode next=cur.next;
cur.next=pre;
pre=cur;
cur=next;
}
return pre;
}