1.本道题的目标是对链表进行深度复制,首先第一种解法是利用map方便操作。旧结点为key,新节点为value。这么做的目的是为了第二遍扫描的时候我们按照这个哈希表把结点的随机指针接上。时间复杂度为O(n),空间复杂度为O(n)。
public RandomListNode copyRandomList(RandomListNode head) {
if(head == null)
return head;
HashMap<RandomListNode,RandomListNode> map = new HashMap<RandomListNode,RandomListNode>();
RandomListNode newHead = new RandomListNode(head.label);
map.put(head,newHead);
RandomListNode pre = newHead;
RandomListNode node = head.next;
while(node!=null)
{
RandomListNode newNode = new RandomListNode(node.label);
map.put(node,newNode);
pre.next = newNode;
pre = newNode;
node = node.next;
}
node = head;
RandomListNode copyNode = newHead;
while(node!=null)
{
copyNode.random = map.get(node.random);
copyNode = copyNode.next;
node = node.next;
}
return newHead;
}
2.方法二是从避免额外的线性空间来考虑,需要一个map的主要原因是新节点的random指向的节点可能还没被创建。想避免使用额外
空间,我们只能通过利用链表原来的数据结构来存储结点。基本思路是这样的,对链表进行三次扫描,第一次扫描对每个结点进行复制,然后把复制出来的新节点接在
原结点的next,也就是让链表变成一个重复链表,就是新旧更替;第二次扫描中我们把旧结点的随机指针赋给新节点的随机指针,因为新结点都跟在旧结点的下一个,所以赋值
比较简单,就是node.next.random = node.random.next,其中node.next就是新结点,因为第一次扫描我们就是把新结点接在旧结点后面。现在我们把结点的随机指针都接好了,
最后一次扫描我们把链表拆成两个,第一个还原原链表,而第二个就是我们要求的复制链表。因为现在链表是旧新更替,只要把每隔两个结点分别相连,对链表进行分割即可。
这个方法总共进行三次线性扫描,所以时间复杂度是O(n)。而这里并不需要额外空间,所以空间复杂度是O(1)。比起上面的方法,这里多做一次线性扫描,但是不需要额外空间,
还是比较值的。实现的代码如下:
public RandomListNode copyRandomList(RandomListNode head) { if(head == null) return head; RandomListNode node = head; while(node!=null) { RandomListNode newNode = new RandomListNode(node.label); newNode.next = node.next; node.next = newNode; node = newNode.next; } node = head; while(node!=null) { if(node.random != null) node.next.random = node.random.next; node = node.next.next; } RandomListNode newHead = head.next; node = head; while(node != null) { RandomListNode newNode = node.next; node.next = newNode.next; if(newNode.next!=null) newNode.next = newNode.next.next; node = node.next; } return newHead; }