问题描述
给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。
思路
非递归版:使用next指针保存head的下一个位置,pre保存为pre的上一个位置(初始化为空),head向后移动,每次移动时令head.next指向pre,之后使用pre保存当前位置,head移动到next的位置重复上述操作,直到head移动到为NULL的位置,此时pre变为了翻转链表的头指针。
递归版:
递归出口:if (head=null || head.next=null )return head;
将head和head.next看做两个部分,对head.next后面的链表进行反转,新链表的头指针为pre
然后让新链表的尾结点即head.next.next指向旧头结点:head.next.next=head
最后让head的后继节点指向空
代码(非递归版)
class Solution {
public ListNode reverseList(ListNode head) {
ListNode pre = null,next = null;
while (head!=null){
next = head.next;
head.next = pre;
pre = head;
head = next;
}
return pre;
}
}
代码(递归版)
class Solution {
public ListNode reverseList(ListNode head) {
if(head == null || head.next == null) {
return head;
}
ListNode nextHead = head.next;
ListNode pre = reverseList(head.next);
nextHead.next = head;
head.next = null;
return pre;
}
}
问题描述
将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
思路
非递归:创建一个新的链表头结点head,之后分别使用p,p1,p2指针指向head,listNode1,listNode2,然后将listNode1和listNode2中指向小的节点放在p的后继节点上,同时p向后移动,p1和p2中指向元素值小的那一个指针也同时向后移。当其中一个链表遍历完时,直接使用p.next指向未遍历完的链表当前遍历指针上,最后返回head.next作为新链表的头指针。
递归版:
递归出口:
if (list1 == null) { return list2; }
if (list2 == null) {return list1; }
将每个链表看做两个部分,首元素和head.next链表;
if(list1.val < list2.val),list1作为新链表的头结点,合并list1.next链表和list2链表;
if(list1.val >= list2.val),list2作为新链表的头结点,合并list2.next链表和list1链表;
代码(非递归)
class Solution {
public ListNode mergeTwoLists(ListNode list1, ListNode list2) {
ListNode p = list1,q = list2;
ListNode newList = new ListNode(-1);
ListNode newP = newList;
while (p!=null && q!=null){
if(p.val>q.val){
newP.next = q;
q = q.next;
newP = newP.next;
}else {
newP.next = p;
p = p.next;
newP = newP.next;
}
}
if(p!=null){
newP.next = p;
}
if(q!=null){
newP.next = q;
}
return newList.next;
}
}
代码(递归)
class Solution {
public ListNode mergeTwoLists(ListNode list1, ListNode list2) {
if (list1 == null) {
return list2;
}
if (list2 == null) {
return list1;
}
if (list1.val < list2.val) {
list1.next = mergeTwoLists(list1.next, list2);
return list1;
} else {
list2.next = mergeTwoLists(list1, list2.next);
return list2;
}
}
}
问题描述
给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。
输入输出样例
示例 1:
输入:head = [1,2,3,4]
输出:[2,1,4,3]
示例 2:
输入:head = []
输出:[]
思路
可以考虑先对前两个元素进行交换,确定好头结点的位置后,再对后面的元素经过两两交换。
交换的思路是:
当p移动到末尾位置或者为null时不再移动
代码
class Solution {
public ListNode swapPairs(ListNode head) {
if(head == null || head.next==null) return head;
ListNode p = head;
ListNode next = head.next;
ListNode pre = null;
p.next = next.next;
next.next = p;
pre = p;
p = p.next;
head = next;
while (p != null && p.next!=null){
next = p.next;
p.next = next.next;
next.next = p;
pre.next = next;
pre = p;
p = p.next;
}
return head;
}
}
问题描述
给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null 。
图示两个链表在节点 c1 开始相交:
题目数据 保证 整个链式结构中不存在环。
思路
两个遍历指针分别指向两个链表开头,如果一个链表走到头则重新指向另外一个指针开头,这样当两个指针再次相遇时就是相交的位置。
代码
public class Solution {
public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
ListNode p = headA;
ListNode q = headB;
while (p!=q){
p = p==null?headB:p.next;
q = q==null?headA:q.next;
}
return p;
}
}
问题描述
给你一个单链表的头节点 head ,请你判断该链表是否为回文链表。如果是,返回 true ;否则,返回 false 。
思路
先使用快慢指针找到链表中点,再把链表切成两半;然后把后半段翻转;最后比较两半是否相等。
代码
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public boolean isPalindrome(ListNode head) {
if(head == null || head.next == null)return true;
ListNode fast=head,slow=head;
while (fast!=null && fast.next!=null){
fast = fast.next.next;
slow = slow.next;
}
slow = reverse(slow);
fast = head;
while (slow!=null && fast!=null){
if(slow.val != fast.val)return false;
slow = slow.next;
fast = fast.next;
}
return true;
}
private ListNode reverse(ListNode head){
if(head == null || head.next==null) return head;
ListNode newHead = reverse(head.next);
head.next.next = head;
head.next = null;
return newHead;
}
}
问题描述
给定一个已排序的链表的头 head , 删除所有重复的元素,使每个元素只出现一次 。返回 已排序的链表 。
输入输出样例
示例 1:
输入:head = [1,1,2]
输出:[1,2]
示例 2:
输入:head = [1,1,2,3,3]
输出:[1,2,3]
问题描述
使用一个next指针每次直到当前指针的下一个位置,如果下一个位置的值等于当前位置的值,则删除next值向的元素(可能有多个连在一起,需要循环删除)
代码
class Solution {
public ListNode deleteDuplicates(ListNode head) {
if(head==null || head.next==null)return head;
ListNode p = head;
ListNode next = null;
while (p!=null){
next = p.next;
while (next!=null && next.val == p.val){
p.next = next.next;
next = p.next;
}
p = p.next;
}
return head;
}
}
问题描述
给定单链表的头节点 head ,将所有索引为奇数的节点和索引为偶数的节点分别组合在一起,然后返回重新排序的列表。
第一个节点的索引被认为是 奇数 , 第二个节点的索引为 偶数 ,以此类推。
请注意,偶数组和奇数组内部的相对顺序应该与输入时保持一致。
你必须在 O(1) 的额外空间复杂度和 O(n) 的时间复杂度下解决这个问题。
输入输出样例
示例 1:
输入: head = [1,2,3,4,5]
输出: [1,3,5,2,4]
示例 2:
输入: head = [2,1,3,5,6,4,7]
输出: [2,3,6,7,1,5,4]
思路
用两个指针p1,p2分别指向偶链表和奇链表的头结点,然后分别用p1.next.next和p2.next.next节点作为p1,p2的后继节点。然后p1,p2向后遍历,当p1,p2其中一个到达尾结点位置时停止遍历,之后直接将偶链表头指针作为p1的后继节点即可。
代码
class Solution {
public ListNode oddEvenList(ListNode head) {
if(head == null || head.next == null || head.next.next==null) return head;
ListNode tailHead = head.next;
ListNode q = head;//偶数节点头指针
ListNode p = tailHead; //技术节点头指针
ListNode q1 = null;
ListNode p1 = null;
while (q.next!=null&&p.next!=null){
q1 = q.next.next;
p1 = p.next.next;
q.next = q1;
p.next = p1;
q = q1;
p = p1;
}
q.next = tailHead;
return head;
}
}
问题描述
给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。
思路
使用一个快指针比慢指针多k+1步,这样当快指针走到结尾时,只需要删除慢指针前面的数即可。
考虑一个特殊位置:如果第n个节点刚好是开头时,返回head.next即可。
代码
class Solution {
public ListNode removeNthFromEnd(ListNode head, int n) {
ListNode p = head;
if(p==null) return null;
for(int i=0;i<=n;i++){//快指针先向前走k+1步
if(p==null) return head.next;//如果走到了空的位置,说明删除的是第一个节点
p = p.next;
}
ListNode pre = head;
while (p!=null){
pre = pre.next;
p = p.next;
}
pre.next = pre.next.next;
return head;
}
}
问题描述
给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表 。
思路
可以使用链表版的归并排序。即将链表从中间分成两个部分,分别对每个部分进行归并排序,之后将两个有序的链表合并在一起。从中间分开可以使用快慢指针,快指针指向结尾的时候,慢指针就指向了中间的位置。需要注意的点是,需要将两个链表从中间断开。
代码
class Solution {
public ListNode sortList(ListNode head) {
if(head == null || head.next == null)return head;
ListNode fast = head.next;
ListNode slow = head;
while (fast!=null && fast.next!=null){//快指针指向结尾或者为空时都结束
fast = fast.next.next;
slow = slow.next;
}
fast = slow.next;//fast重新指向了右边链表的头结点
slow.next = null;//注意要断开两边的链表,否则下次递归时快慢指针会出问题
ListNode listLeft = sortList(head);
ListNode listRight = sortList(fast);
return merge(listLeft,listRight);
}
//合并两个有序的链表
public ListNode merge(ListNode list1,ListNode list2){
if(list1 == null) return list2;
if(list2 == null) return list1;
if(list1.val<list2.val){
list1.next = merge(list1.next,list2);
return list1;
}else {
list2.next = merge(list1,list2.next);
return list2;
}
}
}