题目:请实现一个函数,复制一个复杂链表。

在复杂链表中,每个结点除了有一个next指针指向下一个结点外,还有一个sibling指针指向链表中的任意结点或者nulL

直观解法:

1.遍历链表,复制链表节点,并连接next节点。2.遍历链表,连接sibling节点。对于一个有n个节点的链表,由于定位每个节点的silbing节点都需要从链表的头结点开始经过O(n)步才能实现,因此这种方法的时间复杂度是O(n2)

优化解法:上述方法的时间主要是用在了sibling节点的定位上,考虑优化此步骤。具体的做法是在复制节点的时候,把原节点N和复制出来的N'节点用一个map保存起来,这样在第二部连接sibling结点的时候就可以以O(1)的时间复杂度定位到相应的sibling节点,总体时间复杂度为O(n),空间复杂度也是O(n)

最优算法:

第一步:根据原始链表的每个节点创建对应的复制结点,把复制结点连接在原结点的后面
第二步:设置复制出来的结点的任意结点sibling节点
第三步:把长链表拆分成两个链表

 package Solution;

 public class No26CopyComplexList {

     static class ComplexListNode{
int value;
ComplexListNode next;
ComplexListNode sibling;
public ComplexListNode(){ }
public ComplexListNode(int value,ComplexListNode next,ComplexListNode sibling){
this.value=value;
this.next=next;
this.sibling=sibling;
}
} //把复杂链表中的所有结点进行复制,并连接在原结点的后面
private static void cloneNode(ComplexListNode head){
ComplexListNode node=head;
while(node!=null){
ComplexListNode clonedNode=new ComplexListNode();
clonedNode.value=node.value;
clonedNode.next=node.next;
clonedNode.sibling=null;
node.next=clonedNode;
node=clonedNode.next;
}
}
//设置复制的节点的指向任意结点的连接关系
private static void connectSiblingNodes(ComplexListNode head){
ComplexListNode node=head;
while(node!=null){
ComplexListNode clonedNode=node.next;
if(node.sibling!=null){
clonedNode.sibling=node.sibling.next;
}
node=clonedNode.next;
}
}
//把长链表拆分成两个单独的链表,并返回复制后的链表的头结点
private static ComplexListNode reconnectNodes(ComplexListNode head){
ComplexListNode node=head;
ComplexListNode clonedHead=null;
ComplexListNode clonedNode=null;
if(node!=null){
clonedHead=clonedNode=node.next;
node.next=clonedNode.next;
node=clonedNode.next;
}
while(node!=null){
clonedNode.next=node.next;
clonedNode=node.next;
node.next=clonedNode.next;
node=clonedNode.next;
}
return clonedHead;
}
public static ComplexListNode copyComplexList(ComplexListNode head){
if(head==null)
return null;
cloneNode(head);
connectSiblingNodes(head);
return reconnectNodes(head);
} public static void printComplexList(ComplexListNode head){
ComplexListNode node=head;
int value=0;
ComplexListNode next=null;
ComplexListNode sibling=null;
while(node!=null){
value=node.value;
next=node.next;
sibling=node.sibling;
StringBuilder sb=new StringBuilder("当前节点的值为:").append(value);
if(next!=null)
sb.append(",下一结点的值为:").append(next.value);
else
sb.append(",当前结点为尾节点");
if(sibling!=null)
sb.append(",指向任意结点的值为:").append(sibling.value);
else
sb.append(",没有指向其他结点的任意结点");
System.out.println(sb.toString());
node=node.next;
}
}
public static void main(String[] args) {
ComplexListNode node1=new ComplexListNode();
ComplexListNode node2=new ComplexListNode();
ComplexListNode node3=new ComplexListNode();
ComplexListNode node4=new ComplexListNode();
ComplexListNode node5=new ComplexListNode();
node1.value=1;
node1.next=node2;
node1.sibling=node3;
node2.value=2;
node2.next=node3;
node2.sibling=node5;
node3.value=3;
node3.next=node4;
node3.sibling=null;
node4.value=4;
node4.next=node5;
node4.sibling=node2;
node5.value=5;
node5.next=null;
node5.sibling=null;
printComplexList(node1);
System.out.println("=========================开始复制============================");
printComplexList(copyComplexList(node1));
}
}

剑指offer面试题26:复杂链表的复制的更多相关文章

  1. 剑指offer 面试题35.复杂链表的复制

    时间O(N),空间O(N) /* struct RandomListNode { int label; struct RandomListNode *next, *random; RandomList ...

  2. 剑指offer面试题26-复杂链表的复制

    题目: 请实现函数ComplexListNode* Clone(ComplexListNode* pHead).复制一个复杂链表. 在复杂链表中.每个节点除了一个m_pNext指针指向下一个节点外,另 ...

  3. 剑指Offer - 九度1524 - 复杂链表的复制

    剑指Offer - 九度1524 - 复杂链表的复制2014-02-07 01:30 题目描述: 输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点 ...

  4. 剑指Offer面试题:14.链表的倒数第k个节点

    PS:这是一道出境率极高的题目,记得去年参加校园招聘时我看到了3次,但是每次写的都不完善. 一.题目:链表的倒数第k个节点 题目:输入一个链表,输出该链表中倒数第k个结点.为了符合大多数人的习惯,本题 ...

  5. 剑指Offer:面试题26——复制复杂的链表(java实现)

    问题描述: 输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点). 思路1: 1.先复制链表节点,并用next链接起来. 2.然后对每一个结点去修改 ...

  6. 剑指Offer:面试题16——反转链表(java实现)

    问题描述 定义一个函数,输入一个链表的头结点,反转该链表并输出反转后的链表的头结点.链表结点如下: public class ListNode { int val; ListNode next = n ...

  7. 【剑指offer 面试题15】链表中倒数第K个结点

    思路: 定义两个指针同时指向head,第一个指针先走K-1步,随后二个指针同时移动,当第一个指针到末尾处时,第二个指针所指向的即为倒数第K个结点. #include <iostream> ...

  8. 剑指offer面试题15:链表中倒数第K个节点

    题目:输入一个链表,输出该链表的倒数第K个节点.为了符合大多数人的习惯,本题从1开始计数,即链表尾节点是倒数第一个节点. 解题思路: 解法一:一般情况下,单向链表无法从后一个节点获取到它前面的节点,可 ...

  9. 剑指offer——面试题23:链表中环的入口节点

    函数: ListNode* MeetingNode(ListNode* pHead) { if(pHead==nullptr) return nullptr; ListNode* quickNode= ...

随机推荐

  1. Find out where to contain the smartforms

    Go to table E071 and give smarforms name and it will give the transport req for that. Run SE03, choo ...

  2. unity 动态更新模型透明度

    RaycastHit[] hits; Vector3 normal = transform.position - target.position; hits = Physics.RaycastAll( ...

  3. hbase-bloom filter

    bloom fliter的作用主要用于提升hbase的读性能,但是会牺牲一定的存储空间. 原理: bloom fliter是一种空间效率很高的随机数据结构,初始状态时,bloom filter是一个包 ...

  4. import os, glob, fnmatch--Python os/glob/fnmatch主要函数总结

    auther: Lart date: 2019-01-17 update: 2019-01-18 09:55:36 --- import os, glob, fnmatch 针对某些操作, 官方推荐这 ...

  5. Codeforces Round #436 (Div. 2)D. Make a Permutation! 模拟

    D. Make a Permutation! time limit per test: 2 seconds memory limit per test: 256 megabytes input: st ...

  6. STS中springmvc.xml配置文件

    <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.sp ...

  7. idea运行固定多个模块项目

    第一步:配置workspace.xml 在.idea文件夹下,搜索RunDashboard位置 <component name="RunDashboard"> < ...

  8. 检索html页自生成&nasp;标签,并替换为空(即去掉空格)

    在开发过程中,遇到这样的一种情况,就是页面有时候不知道什么原因会自动生成一些元素,从而打乱自己原有的一些布局. 原html页源代码: 生成后的html源代码: 可以明显看出自动生成了很多   元素,现 ...

  9. 高速上手C++11 14 笔记1

    1 constexpr constexpr关键字可以让已经具备常量返回的函数运用于常量的位置. c++14起可以在函数内部使用局部变量.循环和分支等简单语句. 2 委托构造&继承构造 委托构造 ...

  10. 非交互式一句话添加root用户

    useradd -p `openssl passwd -1 -salt ‘lsof’ admin123` -u 0 -o -g root -G root -s /bin/bash -d /usr/bi ...