0%

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
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
#include <stdio.h>
#include <stdlib.h>
//链表的实现和函数实现
/*链表的结构必定包涵一个指针域,所以地址的分配不一定连续
由计算据随机分配,所以单链表只能单向操作,但是也正是因此
可以使得表基本无线扩容。
结构:
每个数据元素由结点构成
链表的第一个元素节点称为首结点
最后一个元素称为尾结点,指针域NULL
一般定义head(头节点)指向首结点a1,这样比较好!!!
*/
typedef int ElemType;
typedef struct node
{
ElemType data; //数据域
struct node *next; //指针域
} LNode, *LinkList;

//初始化单链表
int InitList_L(LinkList &L)
{
L = (LinkList)malloc(sizeof(LNode)); //分配空间
if (!L)
return 0; //判断是否成功
L->next = NULL; //初始结点指针域为空
return 1;
}

//销毁单链表
int DestroyList(LinkList &L)
{
LinkList p;
while (L)
{
p = L->next;
free(L);
}
}

//求单链表的长度
int ListLength(LinkList L)
{
int i = 0;
//不能直接while(L)因为这样会把头节点移动到尾部;
LinkList p = L->next;
while (p)
{
i++;
p = p->next;
}
return i;
}

//求单链表中值为e的元素
int LocataElem(LinkList L, ElemType e)
{
int i = 0;
LinkList p = L->next;
while (p)
{
i++;
if (p->data == e)
return i;
p = p->next;
}
return 0; //如果没有找到,返回0;
}

//单链表的插入操作
int ListInsert_L(LinkList &L, int i, ElemType e)
{
LinkList p, s; //创建两个指向结点的指针,这里的LinkList等价于LNode * ;
p = L; //让p指向头节点;
int j = 0;
while (p && j < i - 1)
{ //先让p找到目标结点的上一个结点,
p = p->next;
++j;
}
if (!p || j > i - 1)
return 0; //判断i是否合法
//给s分配空间,来存储数据e,
s = (LinkList)malloc(sizeof(LNode));
s->data = e;

/*先让s的指针域指向i,此时i-1对应的结点,此时s和p
都指向目标结点*/
s->next = p->next;

/*再让p指向s,这样就把s插入到目标结点的前面了,也
就是把e插入进去了,此时链表i位置上就是e了;*/
p->next = s;
return 1;
}

//单链表的删除算法,类似于插入操作
int ListDelete_L(LinkList &L, int i, ElemType &e)
{
LinkList p, q;
p = L;
int j = 0;
//同样,先找到i-1个位置
while (p->next && j < i - 1)
{
p = p->next;
++j;
}
//判断上一步是否正确进行
if (!p->next || j > i - 1)
{
return 0;
;
}
/*这里要先让p指向q,再让q指向p->next,
也就是把链表通过q链接起来,再让p指向
q的下一个位置,此时p就直接跳过i结点,指向
i+1结点了,这时候就可以释放i结点了/
如果直接让q指向i+1的结点,那么i的结点就会被遗忘在
缓存区,当程序运行足够长时间时,会导致内存不够,
或者其他的关于内存的问题*/
q = p->next;
p->next = q->next;
e = q->data;
free(q);
return 1;
}

//显示链表的元素和物理存储位置
void ShowAddress(LinkList L)
{
int i;
LinkList p = L->next;
//当然,这里如果使用while写起来会更简便一些
if (p)
{
printf("\n头节点的地址为:\t%p\n", L); //前面说过了,头结点一般不存储数据
for (i = 1; p; i++)
{
printf("第%d个结点的值为:\t%d;\t地址为:%p\n", i, p->data, p);
p = p->next;
}
}
}

/*确定一个链表中元素值最大的结点,我这里设定的是返回对应的结点
当然,也可以返回具体的值*/
LNode *MaxNode(LNode *L)
{
LNode *p = L->next, *maxp = p;
while (p != NULL)
{
if (maxp->data < p->data)
maxp = p;
p = p->next;
}
return maxp;
}
/*int Search_L(LinkList L)//这是返回具体的值函数
{
LinkList p;
ElemType e;
p = L->next;//第一次写的时候这里出现了问题,原因是L指向头结点,里面没有数据
//通过不断的调试程序,我发现每次输出的值的地址和都不一样,所以我认为一定是
//出现了地址和数值比较的情况,回头检查,意识到这里出现了问题。
e = p->data;
p = p->next;
while (p)
{
if (p->data > e)
e = p->data;
p = p->next;
}
return e;
}
*/

//删除元素值最大的结点,这是一个经常用到的,但是比较难的知识点
/*注意:再删除单链表的某一位置的时候,要知道它的前一个位置
1,p遍历,同时pre指向p的前驱
2,用maxp 同步p,使得maxp指向最大值,同样maxpre指向maxp的前驱
3,基于maxpre和maxpre执行删除操作
*/
void DelMaxNode(LNode *&L) //其实这里我觉的参数改成LinkList &L,也行
{
LNode *p = L->next, *pre = L, *maxp = p, *maxpre = pre;
while (p)
{
if (maxp->data < p->data) //比较大小
{
maxp = p;
maxpre = pre;
}
//对pre和maxpre同步后移,保证紧随p,maxp后面
pre = p;
p = p->next;
}
maxpre->next = maxp->next; //简单的删除操作1
free(maxp); //简单的删除操作2
}
/*将表中元素逆置*/
void ListReverse(LNode *&L)
{
//先将表一分为二,一个是空的表头,一个是无表头的表
LNode *p = L->next, *q;
L->next = NULL;
//尾插法实现
while (p)
{
q = p->next;
p->next = L->next;
L->next = p;
p = q; //尾插法这一步特别关键,可以保持p始终在第一个元素的位置(L的后继)
}
}
//查找带头结点的单链表的倒数k个位置上的地址
/*思路:其实和前面的删除最大值的思路一致,只不过此时
的同步指针不是立刻同步,而是在p移动了k个单位之后,才
进行同步*/
int Searchk(LinkList list, int k)
{
LinkList p, q;
int count = 0;
p = q = list->next;
while (p && count < k) //把p从首结点移动到k结点
{
p = p->next;
count++;
}
//判断k的位置是否合法
if (p == NULL)
return 0;
else
{
while (p) //此时只要p指向尾节点,那么q就指向了倒数k的结点,输出就行
{
q = q->next;
p = p->next;
}
printf("倒数%d个位置的元素是:%d", k, q->data); //打印
return 1;
}
}
int main()
{
int A[8] = {1, 2, 3, 4, 5, 100, 200, 300}; //测试数据
int i, j, e = 586; //e时是测试数据
LinkList head, p; //head是头节点,相当于&L;
InitList_L(head);

for (i = 1, j = 0; i <= 8; i++, j++) //注意这里i从1开始,
ListInsert_L(head, i, A[j]); //将数据依次放入
Search_L(head);
printf("%d", Search_L(head));
}