zoukankan      html  css  js  c++  java
  • 【链表问题】打卡8:复制含有随机指针节点的链表

    前言

    以专题的形式更新刷题贴,欢迎跟我一起学习刷题,相信我,你的坚持,绝对会有意想不到的收获。每道题会提供简单的解答,如果你有更优雅的做法,欢迎提供指点,谢谢。

    注:如果代码排版出现了问题麻烦通知我下,谢谢。

    【题目描述】

    【要求】

    如果链表的长度为 N, 时间复杂度达到 O(N)。

    【难度】

    尉:★★☆☆

    【解答】

    方法一:使用额外的存储空间

    这道题的难点在于我们需要定位好随机指针,一个比较简单的解法就是把原节点与复制的节点关联起来,可以使用哈希表把他们关联起来。

    首先把副节点全部创建出来,然后把原节点与对应的副节点用哈希表关联起来。关联的时候原节点作为key,副节点作为value。例如对于链表 1->2->3->null。创建副节点 1', 2', 3'。然后用哈希表关联起来:

    key value
    1 1'
    2 2'
    3 3'

    之后在把所有副节点连接成一个链表。在连接的时候,我们 可以通过哈希表很容易这找到对应的随机节点。

    代码如下

        //方法1:采用哈希表
        public static Node1 copyListWithRand(Node1 head) {
            Map<Node1, Node1> map = new HashMap<>();
            Node1 cur = head;
            while (cur != null) {
                map.put(cur, new Node1(cur.value));
                cur = cur.next;
            }
            //把副节点连接起来
            cur = head;
            while (cur != null) {
                map.get(cur).next = map.get(cur.next);
                map.get(cur).rand = map.get(cur.rand);
                cur = cur.next;
            }
            return map.get(head);
        }
    

    这种方法的时间复杂度为 O(n), 空间复杂度也为 O(n)。

    方法2

    其实我们也可以不需要哈希表来辅助,也就是说 ,我们是可以做到空间复杂度为 O(1)的,我们可以把复制的副节点插入到原链表中去,这样也能把原节点与副节点进行关联,进而
    定位到随机节点。例如,对于链表 1->2->3->null。首先生成副节点 1', 2', 3。然后把副节点插入到原节点的相邻位置,即把原链表变成 1->1'->2->2'->3->3'->null。

    这样我们也可以在连接副节点的时候,找到相应的随机节点。例如 1 的随机节点是 3,则 1' 的随机节点是 3'。显然,1节点的随机节点的下一个节点就是 1'的随机节点。具体代码如下:

        //方法二
        public static Node1 copyListWithRand2(Node1 head){
            Node1 cur = head;
            Node1 next = null;
    
            //把复制的节点插进去
            while (cur != null) {
                next = cur.next;
                Node1 temp = new Node1(cur.value);//复制节点
                temp.next = cur.next;
                cur.next = temp;
                cur = next;
            }
            //在一边把复制的节点取出来一边连接。
            cur = head;
            next = null;
            while (cur != null) {
                next = cur.next.next;//保存原链表的下一个节点
                cur.next.next = next != null ? next.next : null;
                cur.next.rand = cur.rand != null ? cur.rand.next : null;
                cur = next;
            }
            return head.next;
        }
    

    采用这种方法的时候,由于随机节点有可能是空指针,随意写代码的时候要注意。

    问题拓展

    思考:如果是有两个随机指针呢?又该如何处理呢?三个呢?

    【题目描述】

    【要求】

    【难度】

    未知。

    【解答】

    提醒:别想太多了,保持清醒。

    往期

    【链表问题】打卡7:将单向链表按某值划分成左边小,中间相等,右边大的形式

    【链表问题】打卡6:三种方法带你优雅判断回文链表

    【链表问题】环形单链表约瑟夫问题

    最后推广下我的公众号:苦逼的码农,文章都会首发于我的公众号,期待各路英雄的关注交流。

  • 相关阅读:
    牛客 Wannafly 挑战赛26D 禁书目录 排列组合 概率期望
    UOJ#269. 【清华集训2016】如何优雅地求和
    斯特林数 学习笔记
    AtCoder Grand Contest 006 (AGC006) C
    Codeforces 1045D Interstellar battle 概率期望
    Codeforces 1045A Last chance 网络流,线段树,线段树优化建图
    Codeforces 1053C Putting Boxes Together 树状数组
    Codeforces 109D String Transformation 字符串 哈希 KMP
    AtCoder Grand Contest 1~10 做题小记
    AtCoder Grand Contest 002 (AGC002) F
  • 原文地址:https://www.cnblogs.com/kubidemanong/p/10426985.html
Copyright © 2011-2022 走看看