Loading

Leedcode算法专题训练(链表)

1.发现两个链表的交点

160.两个链表的交集(容易)

Leetcode /力扣

public class Solution {
    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
        ListNode curA=headA;
        ListNode curB=headB;
        while(curA!=curB){
            if(curA==null){
                curA=headB;
            }else{
               curA=curA.next; 
            }
            if(curB==null){
                curB=headA;
            }
            else{
                curB=curB.next;
            }  
            
        }
        return curA;
    }
}

2. 链表反转

206. Reverse Linked List (Easy)

Leetcode / 力扣

方法一:迭代

class Solution {
    public ListNode reverseList(ListNode head) {
        ListNode pre=null;
        ListNode 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 next= head.next;
        ListNode cur=reverseList(next);
        next.next=head;
        head.next=null;
        return cur;
    }
}

3. 归并两个有序的链表

21. Merge Two Sorted Lists (Easy)

Leetcode / 力扣

class Solution {
    public ListNode mergeTwoLists(ListNode l1, ListNode l2) {
        ListNode head=new ListNode(0);
        ListNode cur=head;
        while(l1!=null&&l2!=null){
            if(l1.val<l2.val){
                cur.next=l1;
                l1=l1.next;
            }
            else{
                cur.next=l2;
                l2=l2.next;
            }
            cur=cur.next;
        }
        cur.next=l1!=null?l1:l2;
        return head.next;
        
    }
}

 

4. 从有序链表中删除重复节点

83. Remove Duplicates from Sorted List (Easy)

Leetcode / 力扣

 

class Solution {
    public ListNode deleteDuplicates(ListNode head) {
        ListNode cur=head;
        while(cur!=null&&cur.next!=null){
            if(cur.next.val==cur.val){
                cur.next=cur.next.next;
            }
            else cur=cur.next;
        }
        return head;
    }
}

5. 删除链表的倒数第 n 个节点

19. Remove Nth Node From End of List (Medium)

Leetcode / 力扣

这个题目一次遍历的方法挺好的,就是双指针的方式

 

class Solution {
    public ListNode removeNthFromEnd(ListNode head, int n) {
        ListNode first =head;
        while(n-->0){
            first=first.next;
        }
        if(first==null)return head.next;
        ListNode second =head;
        while(first.next!=null){
            first=first.next;
            second=second.next;
        }
        second.next=second.next.next;
        return head;
    }
}

6. 交换链表中的相邻结点

24. Swap Nodes in Pairs (Medium)

Leetcode / 力扣

Given 1->2->3->4, you should return the list as 2->1->4->3.

这题目没做出来需要好好思考,两个指针是可以进行互换的但是需要有个temp节点来记录前一个节点

非递归的解法

class Solution {
    public ListNode swapPairs(ListNode head) {
      	//终止条件:链表只剩一个节点或者没节点了,没得交换了。返回的是已经处理好的链表
        if(head == null || head.next == null){
            return head;
        }
      	//一共三个节点:head, next, swapPairs(next.next)
      	//下面的任务便是交换这3个节点中的前两个节点
        ListNode next = head.next;
        head.next = swapPairs(next.next);
        next.next = head;
      	//根据第二步:返回给上一级的是当前已经完成交换后,即处理好了的链表部分
        return next;
    }
}

递归解法,需要重视的是递归解法需要重视单个递归函数做了什么, 而不是下一层函数做了什么,因此我们只需要关注一级递归的解决过程即可。

因此,也就有了我们解递归题的三部曲:

  1. 找整个递归的终止条件:递归应该在什么时候结束?
  2. 找返回值:应该给上一级返回什么信息?
  3. 本级递归应该做什么:在这一级递归中,应该完成什么任务?

7. 链表求和

445. Add Two Numbers II (Medium)

Leetcode / 力扣

Input: (7 -> 2 -> 4 -> 3) + (5 -> 6 -> 4)
Output: 7 -> 8 -> 0 -> 7

题目要求:不能修改原始链表。

class Solution {
    public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        Stack<Integer> stack1 = new Stack<>();
        Stack<Integer> stack2 = new Stack<>();
        while(l1!=null){
            stack1.push(l1.val);
            l1=l1.next;
        }
        while(l2!=null){
            stack2.push(l2.val);
            l2=l2.next;
        }
        int carry=0;
        ListNode head=null;
        while(!stack1.isEmpty()||!stack2.isEmpty()||carry!=0){
            int a=stack1.isEmpty()?0:stack1.pop();
            int b=stack2.isEmpty()?0:stack2.pop();
            int cur=a+b+carry;
            carry=cur/10;
            cur%=10;
            ListNode curnode =new ListNode(cur);
            curnode.next=head;
            head=curnode;
        }
        return head;
    }
}

8. 回文链表

234. Palindrome Linked List (Easy)

Leetcode / 力扣

题目要求:以 O(1) 的空间复杂度来求解。

切成两半,把后半段反转,然后比较两半是否相等。

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
    public boolean isPalindrome(ListNode head) {
        ListNode cur=head;
        ArrayList<Integer> arr=new ArrayList<>();
        while(cur!=null){
            arr.add(cur.val);
            cur=cur.next;
        }
        int count=arr.size();
        for(int i=0;i<count;i++){
            if(!arr.get(i).equals(arr.get(count-i-1)))return false;
        }
        return true;
    }
}

get()方法对127以内的整形数值会直接封装成值返回,但是对大于127的值会返回对象,因此你的比较是进行对象地址的比较,实际上是不对的。你得用equals()方法比较。

class Solution {
    ListNode temp;
    public boolean isPalindrome(ListNode head) {
        temp=head;
        return check(head);
    }

    private boolean check(ListNode head){
        if(head==null){
            return true;
        }
        boolean res=check(head.next)&&(temp.val==head.val);
        temp=temp.next;
        return res;
    }
}

这个递归太帅了,太有技巧性了

9. 分隔链表

725. Split Linked List in Parts(Medium)

Leetcode / 力扣

Input:
root = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10], k = 3
Output: [[1, 2, 3, 4], [5, 6, 7], [8, 9, 10]]
Explanation:
The input has been split into consecutive parts with size difference at most 1, and earlier parts are a larger size than the later parts.

题目描述:把链表分隔成 k 部分,每部分的长度都应该尽可能相同,排在前面的长度应该大于等于后面的。

class Solution {
    public ListNode[] splitListToParts(ListNode root, int k) {
        ListNode cur=root;
        int count=0;
        while(cur!=null){
            count++;
            cur=cur.next;
        }
        int m=count/k;
        int n=count%k;
        ListNode[] arr=new ListNode[k];
        cur=root;
        for(int i=0;i<k&&cur!=null;i++){
            arr[i]=cur;
            int cursize=m+(n-->0?1:0);
            for(int j=0;j<cursize-1;j++){
                cur=cur.next;
            }
            ListNode next=cur.next;
            cur.next=null;
            cur=next;
        }
        return arr;
    }
}

10. 链表元素按奇偶聚集

328. Odd Even Linked List (Medium)

Leetcode / 力扣

class Solution {
    public ListNode oddEvenList(ListNode head) {
        if(head==null){
            return head;
        }
        ListNode odd=head, even =head.next,evenhead=even;
        while(even!=null&&even.next!=null){
            odd.next=odd.next.next;
            odd=odd.next;
            even.next=even.next.next;
            even=even.next;    
        }
        odd.next=evenhead;
        return head;
    }
}

 

posted @ 2020-11-11 11:22  kopoo  阅读(51)  评论(0)    收藏  举报