剑指offer—— 复杂链表复制

题目:

输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针random指向一个随机节点),请对此链表进行深拷贝,并返回拷贝后的头结点。(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)

第一种方法

思路:

  • (1)将链表中的节点用列表存储
  • (2)然后遍历查找每个节点中random指向列表中的哪一个节点对象,并按照列表顺序存储节点random指向的节点对象的下标。
  • (3)创建新节点列表
  • (4)将新节点中的next和random进行链接。
1587908166698.png

代码:

import java.util.ArrayList;

public class Solution1 {
    public class RandomListNode {
        int label;
        RandomListNode next = null;
        RandomListNode random = null;

        RandomListNode(int label) {
            this.label = label;
        }
    }
    public ArrayList<RandomListNode> toArray(RandomListNode pHead){
        if(pHead==null){
            return null;
        }
        ArrayList<RandomListNode> result=new ArrayList<RandomListNode>();
        RandomListNode key=pHead;
        while(key!=null){
            result.add(key);
            key=key.next;
        }
        return result;
    }
    public int getRandomIndex(ArrayList<RandomListNode> oldNode,RandomListNode key){
        for(int i=0;i<oldNode.size();i++){
            if(oldNode.get(i).equals(key)){
                return i;
            }
        }
        return -1;
    }
    public int[] getRandomList(ArrayList<RandomListNode> oldNode){
        int [] result=new int[oldNode.size()];
        for(int i=0;i<oldNode.size();i++){
            result[i]=this.getRandomIndex(oldNode,oldNode.get(i).random);
            System.out.println(result[i]);
        }
        return result;
    }
    public RandomListNode copyNode(RandomListNode old){
        return new RandomListNode(old.label);
    }
    public ArrayList<RandomListNode> getNewList(ArrayList<RandomListNode> oldNode){
        ArrayList<RandomListNode> newNode=new ArrayList<>();
        for(int i=0;i<oldNode.size();i++){
            newNode.add(this.copyNode(oldNode.get(i)));
        }
        return newNode;
    }
    public RandomListNode getNewLink(ArrayList<RandomListNode> newNode,int [] randomList){

        RandomListNode root=newNode.get(0);
        RandomListNode key=root;
        for(int i=1;i<newNode.size();i++){
            key.next=newNode.get(i);
            key=key.next;
        }
        for(int i=0;i<randomList.length;i++){
            if(randomList[i]!=-1){
                newNode.get(i).random=newNode.get(randomList[i]);
            }
        }
        return root;
    }
    public RandomListNode Clone(RandomListNode pHead)
    {
        //第1步:将原始链表数组化 时间复杂度(n),空间复杂度(n)
        ArrayList<RandomListNode> oldNode=this.toArray(pHead);
        if(oldNode==null){
            return null;
        }
        //第2步:遍历查询random所指向的node 时间复杂度(n^2),空间复杂度(0)
        int [] randomList=this.getRandomList(oldNode);
        //第3步:根据数组重新创建一个新Node数组,next和random都为null 时间复杂度(n),空间复杂度(n)
        ArrayList<RandomListNode> newNode=this.getNewList(oldNode);
        //第4步:链接链表 时间复杂度(n),空间复杂度(0)
        RandomListNode newRoot=this.getNewLink(newNode,randomList);
        return newRoot;
    }
    public RandomListNode create(){
        RandomListNode r1=new RandomListNode(1);
        RandomListNode r2=new RandomListNode(2);
        RandomListNode r3=new RandomListNode(3);
        RandomListNode r4=new RandomListNode(4);
        RandomListNode r5=new RandomListNode(5);
        RandomListNode r6=new RandomListNode(6);
        RandomListNode r7=new RandomListNode(7);
        RandomListNode r8=new RandomListNode(8);
        r1.next=r2;
        r2.next=r3;
        r3.next=r4;
        r4.next=r5;
        r5.next=r6;
        r6.next=r7;
        r7.next=r8;
        r1.random=r3;
        r2.random=r4;
        r3.random=r5;
        r4.random=r6;
        r5.random=r7;
        r6.random=r8;
        r7.random=r1;
        r8.random=r2;
        return r1;
    }

    public static void main(String[] args) {
        Solution1 s=new Solution1();
        RandomListNode root=s.create();
        s.Clone(root);
    }
}

算法分析:

该方法最复杂的地方就是寻找random所指向的节点。

时间复杂度O(n^2),空间复杂度O(n)

代码运行:

运行时间:24ms

占用内存:9656k

第二种方法

这个题目最大的难点在于random引用如何便捷正确的指向对应的新节点。

本方法思路:

  • (1)依次创建新的节点插入到对应的旧节点之后

  • (2)此时新节点的random就在旧节点的random后面,根据这一规律,对新节点的random进行赋值。

  • (3)将组合链表中旧节点删除,可以借助一个额外节点作为暂时的根节点。

    1587969275319.png

代码:

public class Solution2 {
    public class RandomListNode {
        int label;
        RandomListNode next = null;
        RandomListNode random = null;

        RandomListNode(int label) {
            this.label = label;
        }
    }
    public void insertNewNodeInLink(RandomListNode pHead){
        RandomListNode p=pHead;
        while(p!=null){
            RandomListNode q=new RandomListNode(p.label);
            q.next=p.next;
            p.next=q;
            p=q.next;
        }
    }
    public void replaceRandom(RandomListNode pHead){
        RandomListNode t=pHead;
        while(t!=null){
            RandomListNode q=t.next;
            if(t.random!=null)
                q.random=t.random.next;
            t=q.next;

        }
    }
    public RandomListNode getNewLink(RandomListNode pHead){
        RandomListNode s=new RandomListNode(0);
        RandomListNode s1=s;
        while(pHead!=null){
            RandomListNode  q=pHead.next;
            pHead.next=q.next;
            q.next=s.next;
            s.next=q;
            s=s.next;
            pHead=pHead.next;


        }
        return s1.next;
    }
    public RandomListNode Clone(RandomListNode pHead)
    {
        //第一步:遍历创建新的复制节点,并将其插入到旧链表中被复制节点之后 时间复杂度(n),空间复杂度(0)
        this.insertNewNodeInLink(pHead);
        //第二步:因为新节点就在旧节点的后一个节点,因此random所指的旧节点向下移一个,就能够指向新节点 时间复杂度(n),空间复杂度(0)
        this.replaceRandom(pHead);
        //第三步:将新节点串联成新的链表 时间复杂度(n),空间复杂度(0)
        RandomListNode newroot = this.getNewLink(pHead);

        return newroot;

    }
    public RandomListNode create(){
        RandomListNode r1=new RandomListNode(1);
        RandomListNode r2=new RandomListNode(2);
        RandomListNode r3=new RandomListNode(3);
        RandomListNode r4=new RandomListNode(4);
        RandomListNode r5=new RandomListNode(5);
        RandomListNode r6=new RandomListNode(6);
        RandomListNode r7=new RandomListNode(7);
        RandomListNode r8=new RandomListNode(8);
        r1.next=r2;
        r2.next=r3;
        r3.next=r4;
        r4.next=r5;
        r5.next=r6;
        r6.next=r7;
        r7.next=r8;
        r1.random=r3;
        r2.random=r4;
        r3.random=r5;
        r4.random=r6;
        r5.random=r7;
        r6.random=r8;
        r7.random=r1;
        r8.random=r2;
        return r1;

    }

    public static void main(String[] args) {
        Solution2 s=new Solution2();
        RandomListNode root=s.create();
        RandomListNode t=s.Clone(root);

        while(t!=null){
            System.out.println(t.label);
            t=t.next;
        }
    }
}

算法分析

空间复杂度:O(0)

时间复杂度:O(n)

运行代码:

运行时间:19ms

占用内存:9428k

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

推荐阅读更多精彩内容