链表(下)

2018年10月26日

本文主要做一些链表的常见题目,题目从LeetCode上摘取,通过练习加深对链表的掌握和理解。

定义链表的节点类:

    class ListNode {
        int val;
        ListNode next;

        ListNode(int x) {
            val = x;
        }
    }

1,反转链表

题选自LeetCode206题:

反转一个单链表。

示例:
输入: 1->2->3->4->5->NULL
输出: 5->4->3->2->1->NULL

头插法

    public static ListNode reverseList(ListNode head) {
        ListNode prev = null;
        ListNode curr = head;
        while (curr != null) {
            ListNode nextTemp = curr.next;
            curr.next = prev;
            prev = curr;
            curr = nextTemp;
        }
        return prev;
    }

假设反转如下链表:


第一次循环时,currprev为:

第一次循环后各个属性为:


第二次循环时,currprev为:

第二次循环后各个属性为:


第三次循环时,currprev为:

第三次循环后各个属性为:


三次循环后,可以看到把链表反转了,其时间复杂度为 O(N),空间复杂度为 O(1)

递归

    public static ListNode reverseList1(ListNode head) {
        if (head == null || head.next == null) {
            return head;
        }
        //head是p的前一个节点
        ListNode p = reverseList1(head.next);
        //相当于p.next=head
        head.next.next = head;
        //使p的尾节点为null
        head.next = null;
        return p;
    }

其栈的递归调用过程如下:


递归其时间复杂度为 O(N),空间复杂度为 O(N)

2,检测链表中是否有环

题选自LeetCode141题:

给定一个链表,判断链表中是否有环。

如下链表就是有环:


使用HashSet

public boolean hasCycle(ListNode head) {
    Set<ListNode> nodesSeen = new HashSet<>();
    while (head != null) {
        if (nodesSeen.contains(head)) {
            return true;
        } else {
            nodesSeen.add(head);
        }
        head = head.next;
    }
    return false;
}

利用Set中不能有相同元素这一特性,在往nodesSeen集合中添加元素时,一旦有相同元素就返回true,表示有环,若没有环,那么遇到null节点时,会结束循环,并返回false,表示没有环。

使用快慢指针

public boolean hasCycle(ListNode head) {
    if (head == null || head.next == null) {
        return false;
    }
    ListNode slow = head;
    ListNode fast = head.next;
    while (slow != fast) {
        if (fast == null || fast.next == null) {
            return false;
        }
        slow = slow.next;
        fast = fast.next.next;
    }
    return true;
}

试想这么一个场景,甲乙两人绕着标准操场(400m类似椭圆形)跑步,甲的速度比乙快,因为操场时有环的,那么在某一时刻,甲肯定会追上乙,与乙相遇;其中slow表示慢的指针,每次只走一步,而fast表示快的指针,每次走两步,一旦slowfast相等,即它们都指向同一个元素,终止循环,并返回true,表示有环;若fast(偶数情况)或fast.next(奇数情况)指向null,表示这个链表没有环,返回false

3,合并两个有序链表

题选自LeetCode21题:

将两个有序链表合并为一个新的有序链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 

示例:
输入:1->2->4, 1->3->4
输出:1->1->2->3->4->4
    public ListNode mergeTwoLists(ListNode l1, ListNode l2) {
        
        if (l1 == null) {
            return l2;
        }
        if (l2 == null) {
            return l1;
        }
        ListNode listNode = new ListNode(0);
        ListNode curr = listNode;
        while (l1 != null && l2 != null) {
            if (l1.val < l2.val) {
                curr.next = l1;
                l1 = l1.next;
            } else {
                curr.next = l2;
                l2 = l2.next;
            }
            curr = curr.next;
        }
        if (l1 != null) {
            curr.next = l1;
        }
        if (l2 != null) {
            curr.next = l2;
        }
        return listNode.next;
    }

递归

    public ListNode mergeTwoLists(ListNode l1, ListNode l2) {
        if (l1 == null) {
            return l2;
        }
        if (l2 == null) {
            return l1;
        }
        if (l1.val < l2.val) {
            l1.next = mergeTwoLists(l1.next, l2);
            return l1;
        } else {
            l2.next = mergeTwoLists(l1, l2.next);
            return l2;
        }
    }

l1 = 1 -> 2l2 = 1 -> 3为例,其栈递归调用图如下:

4,删除链表的倒数第 n 个节点

题选自LeetCode19题:

给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点。

示例:
给定一个链表: 1->2->3->4->5, 和 n = 2.
当删除了倒数第二个节点后,链表变为 1->2->3->5.

解法1

public ListNode removeNthFromEnd(ListNode head, int n) {
    ListNode dummy = new ListNode(0);
    dummy.next = head;
    int length  = 0;
    ListNode first = head;
    while (first != null) {
        length++;
        first = first.next;
    }
    length -= n;
    first = dummy;
    while (length > 0) {
        length--;
        first = first.next;
    }
    first.next = first.next.next;
    return dummy.next;
}

假设链表的长度为L,那么删除倒数第n个节点,即删除整数第L-n+1个节点,那么就需要获得其前一个节点,即第L-n个节点

解法2

public ListNode removeNthFromEnd(ListNode head, int n) {
    ListNode dummy = new ListNode(0);
    dummy.next = head;
    ListNode first = dummy;
    ListNode second = dummy;
    
    for (int i = 1; i <= n + 1; i++) {
        first = first.next;
    }
    
    while (first != null) {
        first = first.next;
        second = second.next;
    }
    second.next = second.next.next;
    return dummy.next;
}

使用双指针,first在前面跑,因为要铲除倒数第n个节点,那么就要获取到倒数第n+1个节点,所以使firstsecond的距离保持为n+1

5,链表的中间结点

题选自LeetCode876题:

给定一个带有头结点 head 的非空单链表,返回链表的中间结点。
如果有两个中间结点,则返回第二个中间结点。

示例 1:
输入:[1,2,3,4,5]
输出:此列表中的结点 3 (序列化形式:[3,4,5])
返回的结点值为 3 。 (测评系统对该结点序列化表述是 [3,4,5])。
注意,我们返回了一个 ListNode 类型的对象 ans,这样:
ans.val = 3, ans.next.val = 4, ans.next.next.val = 5, 以及 ans.next.next.next = NULL.

示例 2:
输入:[1,2,3,4,5,6]
输出:此列表中的结点 4 (序列化形式:[4,5,6])
由于该列表有两个中间结点,值分别为 3 和 4,我们返回第二个结点。

使用数组

    public ListNode middleNode(ListNode head) {
        ListNode[] A = new ListNode[100];
        int t = 0;
        while (head.next != null) {
            A[t++] = head;
            head = head.next;
        }
        return A[t / 2];
    }

使用快慢指针

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

6,LRU缓存机制

题选自LeetCode146题

运用你所掌握的数据结构,设计和实现一个  LRU (最近最少使用) 缓存机制。它应该支持以下操作: 
获取数据 get 和 写入数据 put 。

获取数据 get(key) - 如果密钥 (key) 存在于缓存中,则获取密钥的值(总是正数),否则返回 -1。
写入数据 put(key, value) - 如果密钥不存在,则写入其数据值。当缓存容量达到上限时,它应该在写入新数据之前删除最近最少使用的数据值,从而为新的数据值留出空间。

进阶:
你是否可以在 O(1) 时间复杂度内完成这两种操作?

示例:

LRUCache cache = new LRUCache( 2 /* 缓存容量 */ );

cache.put(1, 1);
cache.put(2, 2);
cache.get(1);       // 返回  1
cache.put(3, 3);    // 该操作会使得密钥 2 作废
cache.get(2);       // 返回 -1 (未找到)
cache.put(4, 4);    // 该操作会使得密钥 1 作废
cache.get(1);       // 返回 -1 (未找到)
cache.get(3);       // 返回  3
cache.get(4);       // 返回  4

O(N)的解法

public class LRUCache {
    private int capacity;
    private HashMap<Integer, Integer> cacheData;
    private ArrayDeque<Integer> deque;

    public LRUCache(int capacity) {
        this.capacity = capacity;
        cacheData = new HashMap<Integer, Integer>();
        deque = new ArrayDeque<>();
    }

    public int get(int key) {
        if (cacheData.containsKey(key)) {
            deque.remove(key);
            deque.add(key);
            return cacheData.get(key);
        }
        return -1;
    }

    public void put(int key, int value) {
        if (cacheData.containsKey(key)) {
            deque.remove(key);
        }
        if (deque.size() == capacity) {
            cacheData.remove(deque.pollFirst());
        }
        cacheData.put(key, value);
        deque.add(key);
    }
}

因为ArrayDeque中的remove的时间复杂度为O(N),因此总的时间复杂度为O(N)

O(1)解法

public class LRUCacheByList {
    private int size;
    private int capacity;
    private HashMap<Integer, Node> cacheData;
    private Node head;
    private Node tail;

    public LRUCacheByList(int capacity) {
        this.capacity = capacity;
        cacheData = new HashMap<>();
        head = new Node(0, 0);
        tail = new Node(0, 0);
        head.next = tail;
        tail.prev = head;
    }

    public int get(int key) {
        if (cacheData.containsKey(key)) {
            Node node = cacheData.get(key);
            remove(node);
            addLast(node);
            return node.val;
        }
        return -1;
    }

    public void put(int key, int value) {
        if (cacheData.containsKey(key)) {
            Node node = cacheData.get(key);
            node.val = value;
            remove(node);
            addLast(node);
            return;
        }

        Node node = new Node(key, value);
        addLast(node);
        cacheData.put(key, node);
        size++;

        if (size > capacity) {
            cacheData.remove(removeFirst());
            size--;
        }
    }

    private void addLast(Node node) {
        node.prev = tail.prev;
        node.next = tail;

        tail.prev.next = node;
        tail.prev = node;
    }

    private int removeFirst() {
        Node next = head.next;
        Node nextNext = next.next;

        next.prev = null;
        next.next = null;

        nextNext.prev = head;
        head.next = nextNext;

        return next.key;
    }

    private void remove(Node node) {
        Node prev = node.prev;
        Node next = node.next;

        node.prev = null;
        node.next = null;

        prev.next = next;
        next.prev = prev;
    }

    private class Node {
        int key;
        int val;
        Node next;
        Node prev;

        Node(int key, int val) {
            this.key = key;
            this.val = val;
        }
    }
}

上面两种解法思路都是将数据存在HashMap中,使用一个双链表表示数据的“冷热”程度,最新添加的数据从链表尾部插入,最近访问的数据线将其从链表中删除,在将其从链表尾部插入;当空间满了,就删除链表头部的数据;越靠近链表表头,数据越“冷”,越靠近链表尾部,数据越“热”。

上面两种解法都是使用了HashMap与双链表来实现,唯一的区别就是O(N)的解法使用JAVA库中的ArrayDeque来实现双链表;而O(1)解法中自己实现双链表,将节点作为HashMap中的值,当要删除链表中某个节点,通过HashMap取出这个节点,直接更改prevnext即可删除,所以其时间复杂度为O(1)。使用库中的数据结构时,无论是ArrayDeque还是LinkedList,其节点信息都封装在其类里面,无法获取,因此要删除某个节点,只能从头开始遍历,找出与要删除的节点值相同的节点,然后在将其删除,因此其时间复杂度为O(N)

下面将以图解形式分析LRU缓存机制,当执行完以下代码时:

LRUCache cache = new LRUCache( 2 );
cache.put(1, 1);
cache.put(2, 2);

HashMap与链表中的结构如下:

执行:

cache.get(1);       // 返回  1

此时HashMap中没有改动,链表改动如下:

执行:

cache.put(3, 3);    // 该操作会使得密钥 2 作废

此时HashMap与链表改动如下:

执行:

cache.get(2);       // 返回 -1 (未找到)

此时HashMap与链表均没有改动。

执行:

cache.put(4, 4);    // 该操作会使得密钥 1 作废

此时HashMap与链表改动如下:

执行:

cache.get(1);       // 返回 -1 (未找到)

此时HashMap与链表均没有改动。

执行:

cache.get(3);       // 返回  3

此时HashMap中没有改动,链表改动如下:

执行:

cache.get(4);       // 返回  4

此时HashMap中没有改动,链表改动如下:

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

推荐阅读更多精彩内容