编程语言
首页 > 编程语言> > 剑指office算法题--复杂链表的复制

剑指office算法题--复杂链表的复制

作者:互联网

题目

输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的head。(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)

思路

首先要对题目中的复杂链表进行理解,可以根据题目中我们可以理解到这个复杂链表不像我们以前的链表那样,只是单纯的从前指向后面的关系,还包含了一个随机的random指针指向链表中的一个随机节点

randomabc

如上图所示,这就是一个复杂链表,对于节点a他不止只有指向下一个节点b,他还有一个指向节点c的指针random

对于这个题目的解法有两种

方法一

首先我们需要了解map这个Java集合的作用,其中map是一个包含有键值对的数据的集合,关系可以对于为key-value,并且可以通过查找key来查找到value

import java.util.Map;
import java.util.HashMap;
public class Solution {
    public RandomListNode Clone(RandomListNode pHead)
    {
        if(pHead == null)return null;
        RandomListNode newHead = null;//构造一个新的链表的头节点,初始化为null
        RandomListNode p = pHead;//指向当前链表的头节点位置,由于头节点pHead不能随便移动,所以找一个中间变量p来指代头节点的位置
        RandomListNode q = null;//这个q跟上面的p类似,都是指向头节点的中间变量,方便进行链表的节点操作
        Map<RandomListNode, RandomListNode> map = new HashMap<>();//这里用新旧节点来进行一个键值对的映射
        while(p != null){
            if(newHead == null){
                newHead = new RandomListNode(pHead.label);
                q = newHead;
                map.put(pHead, newHead);//上面第一个if里面的判断是看新的链表的头节点如果为空的化,
                //直接将原链表中头节点的值赋给新链表,并且构建新旧链表的键值对映射
            }else{
                if(p.next!= null && map.containsKey(p.next))
                    q.next = map.get(p.next);//这里是如果新链表的头节点不为空,
                    //并且旧链表的下一个节点也不为空并且在map中可以查到该节点的键值对中的key值,
                    //那么就可以直接将value提取出来,该值就是新链表中当前节点的下一个节点的值
                else{
                    if(p.next!= null){
                        RandomListNode temp = new RandomListNode(p.next.label);
                        map.put(p.next, temp);
                        q.next = temp;
                    }//这里是如果旧链表的当前节点的下一个节点不为null,但是map中也找不到该键值对的映射,
                    //那就只有自己进行新链表的赋值,并且从新构建键值对的映射
                }
                if(p.random != null && map.containsKey(p.random))
                    q.random = map.get(p.random);
                else{
                    if(p.random != null){
                        RandomListNode temp = new RandomListNode(p.random.label);
                        map.put(p.random, temp);
                        q.random = temp;    
                    }
                }//这里的操作原理跟上面类似,但是针对的是random指针
                p = p.next;
                q = q.next;
            }
        }
        return newHead;
    }
}

这种方法的优点是可以利用map构建键值对的映射,大大的节省了时间,但是需要另外开辟新的map空间,在牛客网上提交后的结果是
在这里插入图片描述

方法二

  1. 进行原先链表的复制,新复制出来的节点就插在原来节点的后面
aa'bb'cc'
  1. 然后对random指针的复制
randomN_randomaa'bb'cc' randomabc N_randoma'b'c'
public class Solution {
    public RandomListNode Clone(RandomListNode pHead) {
        if(pHead == null) {
            return null;
        }

        RandomListNode currentNode = pHead;
        //1、复制每个结点,如复制结点A得到A1,将结点A1插到结点A后面;
        while(currentNode != null){
            RandomListNode cloneNode = new RandomListNode(currentNode.label);
            RandomListNode nextNode = currentNode.next;
            currentNode.next = cloneNode;
            cloneNode.next = nextNode;
            currentNode = nextNode;
        }

        currentNode = pHead;
        //2、重新遍历链表,复制老结点的随机指针给新结点,如A1.random = A.random.next;
        while(currentNode != null) {
            currentNode.next.random = currentNode.random==null?null:currentNode.random.next;
            currentNode = currentNode.next.next;
        }

        //3、拆分链表,将链表拆分为原链表和复制后的链表
        currentNode = pHead;
        RandomListNode pCloneHead = pHead.next;
        while(currentNode != null) {
            RandomListNode cloneNode = currentNode.next;
            currentNode.next = cloneNode.next;
            cloneNode.next = cloneNode.next==null?null:cloneNode.next.next;
            currentNode = currentNode.next;
        }

        return pCloneHead;
    }
}

这种方法便于理解,但是要对链表进行三次遍历,所以时间复杂度也就更加的大
在这里插入图片描述

总结

通过这个题目就可以掌握到链表的深拷贝这个方法。

标签:RandomListNode,currentNode,office,next,链表,算法,null,节点
来源: https://blog.csdn.net/wuxinmiao123/article/details/105816058