49. 3种方法实现复杂链表的复制[clone of complex linked list]

【题目】

有一个复杂链表,其结点除了有一个next指针指向下一个结点外,还有一个sibling指向链表中的任一结点或者NULL。其结点的C++定义如下:

C++
Code






1
2
3
4
5
6
7

 

// complex node struct

struct ComplexNode

{
    int value;

     ComplexNode *next;

     ComplexNode *sibling;

};

下图是一个含有5个结点的该类型复杂链表。图中实线箭头表示next指针,虚线箭头表示sibling指针。为简单起见,指向NULL的指针没有画出。

请完成函数ComplexNode* Clone(ComplexNode* pHead),以复制一个复杂链表。

分析】

在常见的数据结构上稍加变化,这是一种很新颖的面试题。要在不到一个小时的时间里解决这种类型的题目,我们需要较快的反应能力,对数据结构透彻的理解以及扎实的编程功底。

分为2步:

(1)复制每一个节点,连接next,初步生成【新链表】,时间复杂度为T(n)=O(1)*n=O(n);

(2)连接sibling,假设在【新链表】中定位sibling的时间复杂度为T,则总的时间复杂度为T(n)=T*n;

由此可见,最为关键的是如何在【新链表】中快速定位sibling节点

【方法1】

统计从头结点该sibling节点的步长step。

假设【原始链表】中的某节点N的sibling指向结点S。由于S的位置在链表上有可能在N的前面也可能在N的后面,所以要定位N的位置我们需要从原始链表的头结点开始找。假设从【原始链表】的头结点开始经过step步找到结点S。那么在【新链表】上从头结点开始经过step可以到结点N‘的sibling对应的结点S‘。定位sibling的时间复杂度为T=O(n)
总的时间复杂度O(n2),空间复杂度为O(1)

【方法2】

使用map映射记录<A,A‘>,<B,B‘>,<C,C‘>,这样定位sibling的时间复杂度为T=O(1)
总的时间复杂度O(n),但是空间复杂度为O(n),相当与以O(n)的空间消耗实现了O(n)的时间效率。

【方法3】

根据原始链表的每个结点N,创建对应的N‘。让后将N和N‘连接起来,这样定位sibling的时间复杂度为T=O(1)
,之后重新分成2个链表即可。总的时间复杂度O(n),空间复杂度为O(1)

(1)创建新节点,并连接next。

(2)定位sibling并连接。

(3)链表拆分成两个:把奇数位置的结点链接起来就是原始链表,把偶数位置的结点链接出来就是复制出来的链表。

【代码】

C++
Code






1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89

 

// complex node struct

struct ComplexNode

{
    int value;

     ComplexNode *next;

     ComplexNode *sibling;
};

//Clone all nodes in a complex linked list with pHead,

//and connect all nodes with next link.

void CloneNodes(ComplexNode *pHead)

{
    //  A-A‘-B-B‘-C-C‘

    if(NULL == pHead)

        return;

     ComplexNode *pNode = pHead;

    while(pNode != NULL)

     {

         ComplexNode *pNewNode = new ComplexNode();

         pNewNode->value = pNode->value;

         pNewNode->next = pNode->next;

         pNewNode->sibling = NULL; // init NULL

        // connect pNode and pNewNode

        pNode->next = pNewNode;


        //move to next node

        pNode = pNewNode->next;

     }
}

//Connect sibling nodes in a complex link list

void ConnectSiblingNodes(ComplexNode *pHead)

{
    //  A-A‘-B-B‘-C-C‘

    if(NULL == pHead)

        return;

     ComplexNode *pNode = pHead;

    while(pNode != NULL)

     {

         ComplexNode *pNewNode = pNode->next;

        // connect sibling nodes

        if(pNode->sibling != NULL)

             pNewNode->sibling = pNode->sibling->next;

        else

             pNewNode->sibling = NULL;

     }
}

// Split a complex list into two:

// Reconnect nodes to get the original list, and its cloned list

ComplexNode *ReconnectNodes(ComplexNode *pHead)

{
    //  A-A‘-B-B‘-C-C‘

    if(NULL == pHead)

        return;

     ComplexNode *pNode = pHead;

     ComplexNode *pNewHead = pHead->next;

     ComplexNode *pNewNode = pNewHead;

    while(pNode != NULL)

     {

        // reconnect next nodes for pNode

        pNode->next = pNewNode->next;

        // move pNode to next

        pNode = pNewNode->next;


        // reconnect next nodes for pNewNode

        if(pNode != NULL)

         {

             pNewNode->next = pNode->next;

             pNewNode = pNode->next;

         }

        else

         {

             pNewNode->next = NULL;

             pNewNode = NULL;

         }

     }
    return pNewHead;
}

ComplexNode *Clone(ComplexNode *pHead)
{

     CloneNodes(pHead);

     ConnectSiblingNodes(pHead);

    return ReconnectNodes(pHead);

}

【参考】

http://zhedahht.blog.163.com/blog/static/254111742010819104710337/

http://www.cnblogs.com/kedebug/archive/2012/12/21/2828131.html

时间: 2024-10-21 15:06:42

49. 3种方法实现复杂链表的复制[clone of complex linked list]的相关文章

两种方法求单链表逆序

1 递归,非常easy 代码: #include<iostream> using namespace std; typedef struct node{ int data; struct node * pNext; }Node ,*pNode; void createNode(pNode & pHead){ int temp; scanf("%d",&temp); pNode p,q; bool isFirst = true; while(temp){ if

判断一个链表是否有环的几种方法

一.单链表是否有环 思路分析: 单链表有环,是指单链表中某个节点的next指针域指向的是链表中在它之前的某一个节点,这样在链表的尾部形成一个环形结构.判断链表是否有环,有以下几种方法. 1 // 链表的节点结构如下 2 typedef struct node 3 { 4 int data; 5 struct node *next; 6 } NODE; (1)最常用方法:定义两个指针,同时从链表的头节点出发,一个指针一次走一步,另一个指针一次走两步.如果走得快的指针追上了走得慢的指针,那么链表就是

单链表反转的2种方法

1 public class ReverseDemo { 2 3 /** 4 * 单链表反转的两种方法 5 */ 6 public static void main(String[] args) { 7 Node head =new Node("a"); 8 Node node2=new Node("b"); 9 Node node3=new Node("c"); 10 head.setNext(node2); 11 node2.setNext(

将单链表排序的两种方法

对单链表排序,通常有两种方法.(PS:考察一个程序员的C语言编程功底,通常看他是否能娴熟的操作链表就知道了.) 方法1:将每一个结点保存到额外的数组中,对数组进行排序,然后根据有序的数组重新构建链表. 方法2:直接对链表进行插入排序,但是实现起来比较复杂一些. 显然,方法1最为简单,因为将链式存储L先转化为顺序存储a[],对顺序存储a[]排序,就避免了较为复杂的链接指针操作.一旦对顺序存储a[]排好序后,根据a[]重新构建一个链表是易如反掌的事情. 1. 单链表的定义如下 typedef str

单链表逆序的几种方法

假设单链表数据结构定义如下: struct ListNode { int val; ListNode *next; ListNode(int x) : val(x), next(NULL) {} }; 单链表有一个头指针指向第一个结点,最后一个结点指向NULL 一.最容易想到的方法,新建一个单链表newNode,每次将原先链表的第一个结点放到newNode后 ListNode* reverseList(ListNode* head) { ListNode *newNode = new ListN

将单链表翻转的两种方法

将一个单链表翻转的描述很简单,例如: 输入: NODE1->NODE2->NODE3->NODE4->NODE5->NULL 输出: NODE5->NODE4->NODE3->NODE2->NODE1->NULL 那么,定义单链表如下: (为简单起见,将data字段定义为int, 当然实际应用中data很可能是一个复杂的结构体) typedef struct list_s { int data; struct list_s *next; } li

反转一个链表的两种方法:递归和循环

下面是反转一个链表的两种方法: 一.循环算法 //反转一个链表,循环算法 LinkList Reverse(LinkList& head) { // if(!head) // return head; //此时不用判断head是否为空,如是空的话返回的也是空 LinkList cur = head; LinkList hou; LinkList beh = 0; while (cur) { hou = cur->next; cur->next = beh; beh = cur; cur

50种方法优化SQL Server数据库查询(转载)

原文地址:http://www.cnblogs.com/zhycyq/articles/2636748.html 查询速度慢的原因很多,常见如下几种: 1.没有索引或者没有用到索引(这是查询慢最常见的问题,是程序设计的缺陷) 2.I/O吞吐量小,形成了瓶颈效应. 3.没有创建计算列导致查询不优化. 4.内存不足 5.网络速度慢 6.查询出的数据量过大(可以采用多次查询,其他的方法降低数据量) 7.锁或者死锁(这也是查询慢最常见的问题,是程序设计的缺陷) 8.sp_lock,sp_who,活动的用

20150503 imx257下实现I2C驱动的四种方法

20150503 imx257下实现I2C驱动的四种方法 2015-05-3 Lover雪儿 时间过得好快,转眼间五一假期就即将结束了,假期期间,大家都潇洒的去玩了,徒留辛辛苦苦的程序员还是窝在宿舍乖乖的敲着代码... 好啦,开开玩笑,辛酸史每家都有一大本,还是要积极的面对生活!!! 今天我们的任务是简单的入门linux内核下i2c设备驱动分离的四种写法. 一.一个简单的i2c驱动 和以前的驱动程序不同,i2c驱动分为drv驱动和dev设备驱动两个文件,不懂的可以参考我以前写的<20150313