【LeetCode】Explore.Learn.Linked List

一、 Singly Linked List


  1. Design Linked List

思路:

基础题目,注意在类里设置一个head,tail和length,可有效节约时间和简化代码。

代码:

class MyLinkedList {

    static class Node {
        int value;
        Node next;

        public Node(int value) {
            this.value = value;
            this.next = null;
        }
    }

    private Node head;
    private Node tail;
    private int length;

    /**
     * Initialize your data structure here.
     */
    public MyLinkedList() {
        head = new Node(0);
        tail = new Node(0);
        head.next = tail;
        tail.next = head;
        length = 0;
    }

    /**
     * Get the value of the index-th node in the linked list. If the index is invalid, return -1.
     */
    public int get(int index) {
        if (index > length - 1) return -1;
        if (index == length - 1) return tail.next.value;
        Node p = head;
        while (index-- >= 0) p = p.next;
        return p.value;
    }

    /**
     * Add a node of value val before the first element of the linked list. After the insertion, the new node will be the first node of the linked list.
     */
    public void addAtHead(int val) {
        Node node = new Node(val);
        node.next = head.next;
        head.next = node;
        if (tail.next == head) tail.next = node;
        length++;
    }

    /**
     * Append a node of value val to the last element of the linked list.
     */
    public void addAtTail(int val) {
        Node node = new Node(val);
        node.next = tail;
        tail.next.next = node;
        tail.next = node;
        length++;
    }

    /**
     * Add a node of value val before the index-th node in the linked list. If index equals to the length of linked list, the node will be appended to the end of linked list. If index is greater than the length, the node will not be inserted.
     */
    public void addAtIndex(int index, int val) {
        if (index > length) return;
        if (index == length) {
            addAtTail(val);
            return;
        }
        Node p = head;
        while (index-- > 0) p = p.next;
        Node node = new Node(val);
        node.next = p.next;
        p.next = node;
        length++;
    }

    /**
     * Delete the index-th node in the linked list, if the index is valid.
     */
    public void deleteAtIndex(int index) {
        if (index > length - 1) return;
        Node p = head;
        while (index-- > 0) p = p.next;
        p.next = p.next.next;
        if (p.next == tail) tail.next = p;
        length--;
    }

    public void printList() {
        Node p = head.next;
        while (p != tail) {
            System.out.print(p.value + "  ");
            p = p.next;
        }
        System.out.println();
    }
}

二、 Two Pointer Technique


  1. Linked List Cycle

思路:

设置两个指针,一前一后即可,可使快指针一次走两步,慢指针一次走一步,若存在环,那快指针总会在某一刻追上慢指针。注意,环可能只是链表的一部分,不一定整个链表构成一个环。

代码:

public boolean hasCycle(ListNode head) {
        if (head == null) return false;
        ListNode walker = head;
        ListNode runner = head;
        while (runner.next != null && runner.next.next != null) {
            walker = walker.next;
            runner = runner.next.next;
            if (walker == runner) return true;
        }
        return false;
    }

  1. Linked List Cycle II

思路1:

先说一种比较简单的思路,可以用一个Map存储对象的地址,找到第一个被重复存储的对象即可。

代码:

public ListNode detectCycle(ListNode head) {
        if (head == null) return null;

        HashMap<Integer, ListNode> map = new HashMap<>();
        while (head != null) {
            if (map.containsKey(head.hashCode())) {
                return map.get(head.hashCode());
            } else {
                map.put(head.hashCode(), head);
            }
            head = head.next;
        }
        return null;
    }

思路2:

使用双指针,还是一快一慢,快指针每次走两步,慢指针每次走一步。下图中,链表头是X,环的第一个节点是Y,slow和fast第一次的交点是Z。因为快指针的速度是慢指针的两倍,所以当它们在Z点相遇时,快指针走过的路程为a+b+c+b,慢指针走过的路程为a+b,而2*(a+b)=a+b+c+b,解得a=c,故此时再令fast=head,然后fast变为每次走一步,slow还是每次走一步,当它们再相遇时就一定会在Y处。


代码:

public ListNode detectCycle(ListNode head) {
    ListNode slow = head, fast = head;
    while(fast != null && fast.next != null) {
        fast = fast.next.next;
        slow = slow.next;
        if (slow == fast) {
            while (head != slow) {
                head = head.next;
                slow = slow.next;
            }
            return slow;                
        }
    }           
    return null;
}

  1. Intersection of Two Linked Lists

思路:

常规思路是迭代求出两个链表的长度,然后对齐链表,找到交点即可。一种比较新颖的思路是,两个链表分别迭代,迭代到尾部时,交换链表头,这样就自动抹平了长度差异,具体见代码。

代码:

public ListNode getIntersectionNode(ListNode headA, ListNode headB) {
    //boundary check
    if(headA == null || headB == null) return null;
    
    ListNode a = headA;
    ListNode b = headB;
    
    //if a & b have different len, then we will stop the loop after second iteration
    while( a != b){
        //for the end of first iteration, we just reset the pointer to the head of another linkedlist
        a = a == null? headB : a.next;
        b = b == null? headA : b.next;    
    }
    
    return a;
}

  1. Remove Nth Node From End of List

思路:

使用双指针解决,若要删除的是倒数第k个元素,则让快指针比慢指针先走k步,然后两个指针一起走,当快指针走到尾部时,慢指针就走到了倒数第k个元素。

代码:

public ListNode removeNthFromEnd(ListNode head, int n) {
        ListNode front = head, back = head;
        while (n-- > 0) front = front.next;
        if (front == null) return head.next;
        while (front.next != null) {
            front = front.next;
            back = back.next;
        }
        back.next = back.next.next;
        return head;
    }

三、 Classic Problems


  1. Reverse Linked List

思路:

比较简单,依次把每个元素插入到头指针之后即可。

代码:

public ListNode reverseList(ListNode head) {
        if (head == null) {
            return head;
        }
        ListNode p = head.next;
        head.next = null;
        while (p != null) {
            ListNode pre = p;
            p = p.next;
            pre.next = head;
            head = pre;
        }
        return head;
    }

  1. Remove Linked List Elements

思路:

比较简单,注意前面加一个哨兵头指针处理起来会比较方便。

代码:

public ListNode removeElements(ListNode head, int val) {
        if (head == null || (head.val == val && head.next == null)) return null;

        ListNode pre = head, p = head.next;

        while (p != null) {
            if (p.val == val) {
                pre.next = p.next;
            } else {
                pre = p;
            }

            p = p.next;
        }

        return head;
    }

  1. Odd Even Linked List

思路:

比较简单,不赘述。

代码:

 public ListNode oddEvenList(ListNode head) {
       if (head == null) return null;

        ListNode odd = head, even = head.next, evenHead = even;

        while (even != null && even.next != null) {
            odd.next = odd.next.next;
            even.next = even.next.next;
            odd = odd.next;
            even = even.next;
        }
        odd.next = evenHead;

        return head;
    }

  1. Palindrome Linked List

思路:

这道题目还是比较有意思的,综合应用了前面的技巧。首先,使用快慢指针,使得慢指针停留在序列中间,然后使用链表反转慢指针到结尾的部分,然后比较这两部分是否相等即可。

代码:

    public boolean isPalindrome(ListNode head) {
        ListNode fast = head, slow = head;
        while (fast != null && fast.next != null) {
            fast = fast.next.next;
            slow = slow.next;
        }
        if (fast != null) { // odd nodes: let right half smaller
            slow = slow.next;
        }
        slow = reverse(slow);
        fast = head;

        while (slow != null) {
            if (fast.val != slow.val) {
                return false;
            }
            fast = fast.next;
            slow = slow.next;
        }
        return true;
    }

    public ListNode reverse(ListNode head) {
        ListNode prev = null;
        while (head != null) {
            ListNode next = head.next;
            head.next = prev;
            prev = head;
            head = next;
        }
        return prev;
    }

四、 Conclusion


  1. Merge Two Sorted Lists

思路:

比较简单,不赘述

代码:

    public ListNode mergeTwoLists(ListNode l1, ListNode l2) {
        ListNode head = new ListNode(-1);
        ListNode p = head;

        while (l1 != null && l2 != null) {

            if (l1.val < l2.val) {
                p.next = l1;
                l1 = l1.next;
            } else {
                p.next = l2;
                l2 = l2.next;
            }
            p = p.next;
        }

        while (l1 != null) {
            p.next = l1;
            l1 = l1.next;
            p = p.next;
        }

        while (l2 != null) {
            p.next = l2;
            l2 = l2.next;
            p = p.next;
        }

        return head.next;
    }

  1. Add Two Numbers

思路:

比较简单,不赘述。

代码:

public static ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        ListNode head = l1;
        int carry = 0;
        int sum = 0;

        while (l1.next != null && l2.next != null) {
            sum = l1.val + l2.val + carry;
            l1.val = sum % 10;
            carry = sum / 10;
            l1 = l1.next;
            l2 = l2.next;
        }

        if (l1.next == null) {
            l1.next = l2.next;
        }

        sum = l1.val + l2.val + carry;
        l1.val = sum % 10;
        carry = sum / 10;

        ListNode p = l1;
        l1 = l1.next;

        while (l1 != null) {
            sum = l1.val + carry;
            l1.val = sum % 10;
            carry = sum / 10;
            l1 = l1.next;
            p = p.next;
        }

        if (carry == 1) {
            p.next = new ListNode(carry);
        }

        return head;
    }

  1. Flatten a Multilevel Doubly Linked List

思路:

很显然用递归解决会比较省力气,当检测到当前Node有child时,就递归进入下一层双向链表,返回值是最后一个链表元素,然后把他们串起来即可。

代码:

public Node flatten(Node head) {
        if (head != null) {
            Node p = recursive(head);
        }
        return head;
    }

    private Node recursive(Node head) {
        Node cur = head;
        while (cur.next != null) {
            if (cur.child != null) {
                Node p = recursive(cur.child);
                p.next = cur.next;
                cur.next.prev = p;
                cur.child.prev = cur;
                cur.next = cur.child;
                cur.child = null;
            }
            cur = cur.next;
        }

        if (cur.child != null) {
            Node p = recursive(cur.child);
            cur.next = cur.child;
            cur.child.prev = cur;
            cur.child = null;
        }
        return cur;
    }

  1. Copy List with Random Pointer

思路:

这道链表的深度拷贝题的难点就在于如何处理随机指针的问题,由于每一个节点都有一个随机指针,这个指针可以为空,也可以指向链表的任意一个节点,如果我们在每生成一个新节点给其随机指针赋值时,都要去遍历原链表的话,OJ上肯定会超时,所以我们可以考虑用Hash map来缩短查找时间,第一遍遍历生成所有新节点时同时建立一个原节点和新节点的哈希表,第二遍给随机指针赋值时,查找时间是常数级。

代码:

public RandomListNode copyRandomList(RandomListNode head) {

        if (head == null) return null;

        HashMap<RandomListNode, RandomListNode> map = new HashMap<>();

        RandomListNode node = head;
        while (node != null) {
            map.put(node, new RandomListNode(node.label));
            node = node.next;
        }

        node = head;
        while (node != null) {
            map.get(node).next = map.get(node.next);
            map.get(node).random = map.get(node.random);
            node = node.next;
        }

        return map.get(head);

    }

  1. Rotate List

思路:

注意到k的值可能比链表长度大,所以要先求链表长度length,然后k=k%length,在让指针走到length - k - 1处即可。

代码:

public static ListNode rotateRight(ListNode head, int k) {

        if (head == null) return head;

        int length = 0;
        ListNode p = head;

        while (p.next != null) {
            length++;
            p = p.next;
        }
        p.next = head;
        length++;
        k = k % length;
        k = length - k - 1;

        p = head;
        while (k-- > 0) {
            p = p.next;
        }

        head = p.next;
        p.next = null;

        return head;
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容