C语言之复杂链表的复制方法(图示详解)

2025-05-27 0 46

什么是复杂链表

复杂链表指的是一个链表有若干个结点,每个结点有一个数据域用于存放数据,还有两个指针域,其中一个指向下一个节点,还有一个随机指向当前复杂链表中的任意一个节点或者是一个空结点。今天我们要实现的就是对这样一个复杂链表复制产生一个新的复杂链表

复杂链表的数据结构如下:

?

1

2

3

4

5

6

7

8

9

10

11
typedef int DataType; //数据域的类型

//复杂链表的数据结构

typedef struct ComplexNode

{

DataType _data ; // 数据

struct ComplexNode * _next; // 指向下个节点的指针

struct ComplexNode * _random; // 指向随机节点(可以是链表中的任意节点 or 空)

}ComplexNode;

C语言之复杂链表的复制方法(图示详解)

上图就是一个复杂链表的例子,那么我们应该如何实现复杂链表复制呢?

1、首先我们应该根据已有的复杂链表创建一条新的复杂链表,但是这个新的复杂链表的所有的结点的random指针都指向空,这样是很好实现的,相当于我们创建了一条简单的单链表(newlist),我们要复制的链表不妨称之为oldlist。

C语言之复杂链表的复制方法(图示详解)

2、接下来我们应该把新创建的这条复杂链表(newlist)与已有的复杂链表(oldlist)合并成如下的形式:

C语言之复杂链表的复制方法(图示详解)

在这种情况下我们已经把两条复杂链表合并成了一条链表(称之为linklist),通过对这条链表(linklist)的观察,我们可以发现合并的链表(linklist)中属于newlist的结点pnew的上一个结点pold(属于oldlist的结点)的random指针所指向的结点的next指针就应该是pnew结点的randow指针所指向的结点。

这样我们让pold和pnew指针一直往后走最后就可以实现对所有属于新创建的复杂链表(newlist)的random指针指向相应的结点的操作。构成的复杂链表如下图

C语言之复杂链表的复制方法(图示详解)

在完成以上的步骤之后我们所要做的工作就很简单了,我们只要把这一条链表linklist分开成我们的newlist链表和oldlist链表就可以了。

C语言之复杂链表的复制方法(图示详解)

C语言之复杂链表的复制方法(图示详解)

这样我们就完美的完成了复杂链表复制工作下面就是具体实现的代码:

头文件complexnode.h:

?

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
#ifndef __COMPLEX__NODE__H__

#define __COMPLEX__NODE__H__

//包含头文件

#include <stdio.h>

#include<stdlib.h>

#include <assert.h>

typedef int DataType; //数据域的类型

//复杂链表的数据结构

typedef struct ComplexNode

{

DataType _data ; // 数据

struct ComplexNode * _next; // 指向下个节点的指针

struct ComplexNode * _random; // 指向随机节点(可以是链表中的任意节点 or 空)

}ComplexNode;

//创建一个复杂链表的结点

ComplexNode * BuyComplexNode(DataType x);

//打印复杂的单链表

void Display(const ComplexNode * cplist);

//复杂链表的复制

ComplexNode * CopyComplexNode(ComplexNode * cplist);

#endif//__COMPLEX__NODE__H__

具体功能实现complexnode.c

?

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

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

204

205

206

207

208

209

210

211

212

213

214

215

216

217

218

219

220

221

222

223
#include "complexnode.h"

//创建一个复杂链表的结点

ComplexNode * BuyComplexNode(DataType x)

{

ComplexNode *cnode = (ComplexNode *)malloc(sizeof(ComplexNode));

if(cnode == NULL)//创建失败

{

perror("BuyComplexNode()::malloc");

return NULL;

}

//创建成功

cnode->_data = x;

cnode->_next = NULL;

cnode->_random = NULL;

return cnode;

}

//打印复杂的单链表

void Display(const ComplexNode * cplist)

{

ComplexNode *pnode = cplist;

while (pnode)

{

printf("%d::%d -->",pnode->_data,pnode->_random->_data);

pnode = pnode->_next;

}

printf("over\\n");

}

//复杂链表的复制

ComplexNode * CopyComplexNode(ComplexNode * cplist)

{

ComplexNode * pold = NULL;

ComplexNode * pnew = NULL;

ComplexNode * newlist = NULL;//指向新的复杂链表的头结点的指针

pold = cplist;

//创建一条新的复杂链表

while(pold != NULL)

{

ComplexNode * new_node = BuyComplexNode(pold->_data);

if(newlist == NULL)//当新的复杂链表中没有结点时

{

newlist = new_node;

}

else//当新的复杂链表有结点时

{

ComplexNode * node = newlist;

while(node->_next != NULL)//找到最后一个结点

{

node = node->_next;

}

node->_next = new_node;//插入新的结点

}

pold = pold->_next;

}//创建新的复杂链表结束

//合并两条复杂链表

pold = cplist;

pnew = newlist;

while (pold)

{

ComplexNode * curold = NULL;

ComplexNode * curnew = NULL;

curold = pold->_next;

curnew = pnew->_next;

if(pold->_next == NULL)

{

pold->_next = pnew;

pold = curold;

pnew = curnew;

break;

}

pold->_next = pnew;

pnew->_next = curold;

pold = curold;

pnew = curnew;

}//合并两条复杂链表结束

//让新创建的那条复杂链表上的所有结点的random指针指向相应的结点

pold = cplist;

pnew = newlist;

while (pnew)

{

pnew->_random = pold->_random->_next;

pold = pnew->_next;

if(pold == NULL)//这是pnew的_next指针已经指向空

{

break;

}

pnew = pold->_next;

}//结束

//分离合并后的复杂链表

pold = cplist;

pnew = newlist;

while (pold)

{

ComplexNode * curold = NULL;

ComplexNode * curnew = NULL;

if(pnew->_next == NULL)//已经分离完成

{

pold->_next = NULL;

pnew->_next = NULL;

break;

}

curold = pold->_next->_next;

curnew = pnew->_next->_next;

pold->_next = curold;

pnew->_next = curnew;

pold = curold;

pnew = curnew;

}//分离合并的复杂链表结束

return newlist;

}

测试代码test.c:

#include "complexnode.h"

//

//复杂链表的复制。?个链表的每个节点,有?个指向next指针指向下?个节

//点,还有?个random指针指向这个链表中的?个随机节点或者NULL,现在要

//求实现复制这个链表,返回复制后的新链表。

//ps: 复杂链表的结构

void test()

{

ComplexNode * cplist;

ComplexNode * copylist;

ComplexNode * node1;

ComplexNode * node2;

ComplexNode * node3;

ComplexNode * node4;

cplist = BuyComplexNode(1);

node1 = BuyComplexNode(2);

node2 = BuyComplexNode(3);

node3 = BuyComplexNode(4);

node4 = BuyComplexNode(5);

cplist->_next = node1;

node1->_next = node2;

node2->_next = node3;

node3->_next = node4;

cplist->_random = node3;

node1->_random = node4;

node2->_random = cplist;

node3->_random = node1;

node4->_random = node2;

Display(cplist);

copylist = CopyComplexNode(cplist);

Display(copylist);

}

int main()

{

test();

return 0;

}

程序的运行结果如下图:

C语言之复杂链表的复制方法(图示详解)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持快网idc。

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 C语言之复杂链表的复制方法(图示详解) https://www.kuaiidc.com/73903.html

相关文章

发表评论
暂无评论