定义:链表是一种物理储存上非连续,数据元素的逻辑顺序通过链表中的指针链接次序,实现的一种现行储存结构。
特点:
链表由一系列节点(链表中每一个元素称为节点)组成,节点在运行时动态生成(malloc),每个节点包括两个部分:
(1)存储数据元素的数据域
(2)存储下一个节点地址的指针域
链表的构成:链表由一个个节点构成,每个节点一般采用结构体的形式组织,例如:
typedef struct student
{
int num;
char name[20];
struct student *next; //与结构体名字一样
}STU;
链表节点分为两个域
数据域:存放实际的数据,如:num,score;
指针域:存放下一个节点的首地址,如:next等。
链表的操作:创建(增删改查)
1.链表的创建
第一步:创建一个节点
第二步:创建第二个节点,将其放在第一个节点的后面(第一个节点的指针域保存第二个节点的地址)
第三步:再次创建节点,找到原本节点的最后一个节点,接着将最后一个节点的指针域保存新节点的地址,以此类推。
代码1:
#include<stdio.h>
#include<stdlib.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* next;
}STU;
void link_creat_head(STU** p_head, STU *p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->next = NULL;
}
}
int main()
{
STU* head = NULL, * p_new = NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for(i=0;i<num;i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_creat_head(&head, p_new); //将节点加入新的链表
}
}
链表的遍历:
第一步:输出第一个结点的数据域,输出完毕后,让指针保存后一个结点的地址
p=p->next
第二步:输出移动地址对应的结点的数据域,输出完毕后,指针继续后移
第三步:以此类推,直到结点的指针域为NULL
代码2:
#include<stdio.h>
#include<stdlib.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* next;
}STU;
void link_creat_head(STU** p_head, STU* p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->next = NULL;
}
}
//链表的遍历
void link_print(STU* head)
{
STU* p_mov;
//定义新的指针保存链表的首地址,防止使用head改变原本链表
p_mov = head;
//当指针保存最后一个结点的指针域为NULL时,循环结束
while (p_mov != NULL)
{
//先打印当前指针保存结点的指针域
printf("num=%d score=%d name=%s \n", p_mov->num, p_mov->score, p_mov->name);
//指针后移 保存下一个结点的地址
p_mov = p_mov->next;
}
}
int main()
{
STU* head = NULL, * p_new = NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for (i = 0; i < num; i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_creat_head(&head, p_new); //将节点加入新的链表
}
link_print(head);
}
链表的释放:
重新定义一个指针q,保存p指向的地址,然后p后移保存下一个结点的地址,然后释放q对应的结点,以此类推,直到p为NULL为止。
代码3:
#include<stdio.h>
#include<stdlib.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* next;
}STU;
void link_creat_head(STU** p_head, STU* p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->next = NULL;
}
}
//链表的遍历
void link_print(STU* head)
{
STU* p_mov;
//定义新的指针保存链表的首地址,防止使用head改变原本链表
p_mov = head;
//当指针保存最后一个结点的指针域为NULL时,循环结束
while (p_mov != NULL)
{
//先打印当前指针保存结点的指针域
printf("num=%d score=%d name=%s \n", p_mov->num, p_mov->score, p_mov->name);
//指针后移 保存下一个结点的地址
p_mov = p_mov->next;
}
}
//链表的释放
void link_free(STU** p_head)
{
//定义一个指针变量保存头指针的地址
STU* pb = *p_head;
while (*p_head!=NULL)
{
//先保存p_head指向的结点的地址
pb = *p_head;
//p_head保存下一个结点的地址
*p_head = (*p_head)->next;
//释放结点并防止野指针
free(pb);
pb = NULL;
}
}
int main()
{
STU* head = NULL, * p_new = NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for (i = 0; i < num; i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_creat_head(&head, p_new); //将节点加入新的链表
}
link_print(head);
link_free(&head);
}
链表结点的查找
先对比第一个结点的数据域是否是想要的数据,如果是就直接返回,如果不是则继续查找下一个结点,如果到达最后一个结点就没有匹配的数据,说明要查找的数据不存在。
代码4:
#include<stdio.h>
#include<stdlib.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* next;
}STU;
void link_creat_head(STU** p_head, STU* p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->next = NULL;
}
}
//链表的遍历
void link_print(STU* head)
{
STU* p_mov;
//定义新的指针保存链表的首地址,防止使用head改变原本链表
p_mov = head;
//当指针保存最后一个结点的指针域为NULL时,循环结束
while (p_mov != NULL)
{
//先打印当前指针保存结点的指针域
printf("num=%d score=%d name=%s \n", p_mov->num, p_mov->score, p_mov->name);
//指针后移 保存下一个结点的地址
p_mov = p_mov->next;
}
}
//链表的释放
void link_free(STU** p_head)
{
//定义一个指针变量保存头指针的地址
STU* pb = *p_head;
while (*p_head!=NULL)
{
//先保存p_head指向的结点的地址
pb = *p_head;
//p_head保存下一个结点的地址
*p_head = (*p_head)->next;
//释放结点并防止野指针
free(pb);
pb = NULL;
}
}
//链表的查找
//按照学号查找
STU* link_search_num(STU* head, int num)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (p_mov->num == num) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
//按照姓名查找
STU* link_search_name(STU* head, char *name)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (strcmp(p_mov->name,name) ==0) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
int main()
{
STU* head = NULL, * p_new = NULL, * pb=NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for (i = 0; i < num; i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_creat_head(&head, p_new); //将节点加入新的链表
}
link_print(head);
//link_free(&head);
//查学号
while (1)
{
printf("输入您要查找的学生的学号:\n");
scanf("%d", &num);
pb = link_search_num(head, num);
if (pb != NULL)
{
printf("找到了 num=%d score=%d name=%s", pb->num, pb->score, pb->name);
}
else
{
printf("没有找到");
}
}
}
链表结点的删除:
如果链表为空,不需要删除
如果删除的是第一个结点,则需要保存链表首地址的指针保存第一个结点的下一个结点的地址
如果删除的是中间结点,则找到中间结点的前一个结点,让前一个结点的指针域保存这个结点的后一个地址即可。
代码5:
#include<stdio.h>
#include<stdlib.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* next;
}STU;
void link_creat_head(STU** p_head, STU* p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->next = NULL;
}
}
//链表的遍历
void link_print(STU* head)
{
STU* p_mov;
//定义新的指针保存链表的首地址,防止使用head改变原本链表
p_mov = head;
//当指针保存最后一个结点的指针域为NULL时,循环结束
while (p_mov != NULL)
{
//先打印当前指针保存结点的指针域
printf("num=%d score=%d name=%s \n", p_mov->num, p_mov->score, p_mov->name);
//指针后移 保存下一个结点的地址
p_mov = p_mov->next;
}
}
//链表的释放
void link_free(STU** p_head)
{
//定义一个指针变量保存头指针的地址
STU* pb = *p_head;
while (*p_head!=NULL)
{
//先保存p_head指向的结点的地址
pb = *p_head;
//p_head保存下一个结点的地址
*p_head = (*p_head)->next;
//释放结点并防止野指针
free(pb);
pb = NULL;
}
}
//链表的查找
//按照学号查找
STU* link_search_num(STU* head, int num)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (p_mov->num == num) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
//按照姓名查找
STU* link_search_name(STU* head, char *name)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (strcmp(p_mov->name,name) ==0) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
//链表结点的删除
void link_detele_num(STU** p_head, int num)
{
STU* pb, * pf;
pb = pf = *p_head;
if (*p_head == NULL) //链表为空 不用删
{
printf("链表为空,没有您要删的结点");
return;
}
while (pb->num!=num && pb->next!=NULL) //循环找,要删除的结点
{
pf = pb;
pb = pb->next;
}
if (pb->num == num) //找到了一个结点的num和num相同
{
if (pb == *p_head) //要删除的是头结点
{
//让保存头结点的指针保存后一个结点的地址
*p_head = pb->next;
}
else
{
//前一个结点的指针域保存要删除的后一个结点的地址
pf->next = pb->next;
}
//释放空间
free(pb);
pb = NULL;
}
else //没有找到
{
printf("没有你要删除的结点\n");
}
}
int main()
{
STU* head = NULL, * p_new = NULL, * pb=NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for (i = 0; i < num; i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_creat_head(&head, p_new); //将节点加入新的链表
}
link_print(head);
//link_free(&head);
printf("请输入你要删除结点的学号:\n");
scanf("%d", &num);
link_detele_num(&head, num);
link_print(head);
link_free(&head);
}
链表中插入一个节点:
链表中插入一个结点,按照原本链表的顺序插入,找到合适的位置。(原本的链表必须有顺序)
情况:
1.如果链表没有结点,则新插入的结点就是第一个结点
2.如果新插入的结点的数值最小 则为头结点
3.如果新插入的数值在中间位置 则找到前一个 然后插入他们中间
4.如果新插入的数值最大 则插入到最后
代码6:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* next;
}STU;
void link_creat_head(STU** p_head, STU* p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->next = NULL;
}
}
//链表的遍历
void link_print(STU* head)
{
STU* p_mov;
//定义新的指针保存链表的首地址,防止使用head改变原本链表
p_mov = head;
//当指针保存最后一个结点的指针域为NULL时,循环结束
while (p_mov != NULL)
{
//先打印当前指针保存结点的指针域
printf("num=%d score=%d name=%s \n", p_mov->num, p_mov->score, p_mov->name);
//指针后移 保存下一个结点的地址
p_mov = p_mov->next;
}
}
//链表的释放
void link_free(STU** p_head)
{
//定义一个指针变量保存头指针的地址
STU* pb = *p_head;
while (*p_head != NULL)
{
//先保存p_head指向的结点的地址
pb = *p_head;
//p_head保存下一个结点的地址
*p_head = (*p_head)->next;
//释放结点并防止野指针
free(pb);
pb = NULL;
}
}
//链表的查找
//按照学号查找
STU* link_search_num(STU* head, int num)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (p_mov->num == num) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
//按照姓名查找
STU* link_search_name(STU* head, char* name)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (strcmp(p_mov->name, name) == 0) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
//链表结点的删除
void link_detele_num(STU** p_head, int num)
{
STU* pb, * pf;
pb = pf = *p_head;
if (*p_head == NULL) //链表为空 不用删
{
printf("链表为空,没有您要删的结点");
return;
}
while (pb->num != num && pb->next != NULL) //循环找,要删除的结点
{
pf = pb;
pb = pb->next;
}
if (pb->num == num) //找到了一个结点的num和num相同
{
if (pb == *p_head) //要删除的是头结点
{
//让保存头结点的指针保存后一个结点的地址
*p_head = pb->next;
}
else
{
//前一个结点的指针域保存要删除的后一个结点的地址
pf->next = pb->next;
}
//释放空间
free(pb);
pb = NULL;
}
else //没有找到
{
printf("没有你要删除的结点\n");
}
}
//链表的插入:按照学号的顺序插入
void link_insert_num(STU** p_head, STU* p_new)
{
STU* pb, * pf;
pb = pf = *p_head;
if (*p_head == NULL) //链表为空
{
*p_head = p_new;
p_new->next = NULL;
return;
}
while ((p_new->num >= pb->num) && (pb->next != NULL))
{
pf = pb;
pb = pb->next;
}
if (p_new->num < pb->num) //找到一个结点的num比新来的结点大 插在pb前
{
if (pb == *p_head) //找到的节点是头结点 插在最前
{
p_new->next = *p_head;
*p_head = p_new;
}
else
{
pf->next = p_new;
p_new->next = pb;
}
}
else //没有找到pb的num比p_new->num大的结点 插在最后面
{
pb->next = p_new;
p_new->next = NULL;
}
}
int main()
{
STU* head = NULL, * p_new = NULL, * pb = NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for (i = 0; i < num; i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_creat_head(&head, p_new); //将节点加入新的链表
}
link_print(head);
//link_free(&head);
while (1)
{
printf("请输入要插入的结点num score name \n");
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_insert_num(&head, p_new);
link_print(head);
}
}
结果:
请输入链表初始个数:3
请输入学号,分数,名字:
1 56 k1
请输入学号,分数,名字:
3 78 k2
请输入学号,分数,名字:
5 98 k3
num=1 score=56 name=k1
num=3 score=78 name=k2
num=5 score=98 name=k3
请输入要插入的结点num score name
4 99 k4
num=1 score=56 name=k1
num=3 score=78 name=k2
num=4 score=99 name=k4
num=5 score=98 name=k3
请输入要插入的结点num score name
代码7:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* next;
}STU;
void link_creat_head(STU** p_head, STU* p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->next = NULL;
}
}
//链表的遍历
void link_print(STU* head)
{
STU* p_mov;
//定义新的指针保存链表的首地址,防止使用head改变原本链表
p_mov = head;
//当指针保存最后一个结点的指针域为NULL时,循环结束
while (p_mov != NULL)
{
//先打印当前指针保存结点的指针域
printf("num=%d score=%d name=%s \n", p_mov->num, p_mov->score, p_mov->name);
//指针后移 保存下一个结点的地址
p_mov = p_mov->next;
}
}
//链表的释放
void link_free(STU** p_head)
{
//定义一个指针变量保存头指针的地址
STU* pb = *p_head;
while (*p_head != NULL)
{
//先保存p_head指向的结点的地址
pb = *p_head;
//p_head保存下一个结点的地址
*p_head = (*p_head)->next;
//释放结点并防止野指针
free(pb);
pb = NULL;
}
}
//链表的查找
//按照学号查找
STU* link_search_num(STU* head, int num)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (p_mov->num == num) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
//按照姓名查找
STU* link_search_name(STU* head, char* name)
{
STU* p_mov;
//定义一个指针变量保存第一个结点的地址
p_mov = head;
//当没有到达最后一个结点的指针域时循环继续
while (p_mov != NULL)
{
//如果找到是当前结点的数据,则返回当前结点的地址
if (strcmp(p_mov->name, name) == 0) //找到了
{
return p_mov;
}
//如果没有找到 则继续对比下一个结点的指针域
p_mov = p_mov->next;
}
//当循环结束的时候还没有找到 说明要查找的数据不存在 返回NULL进行标识
return NULL; //没有找到
}
//链表结点的删除
void link_detele_num(STU** p_head, int num)
{
STU* pb, * pf;
pb = pf = *p_head;
if (*p_head == NULL) //链表为空 不用删
{
printf("链表为空,没有您要删的结点");
return;
}
while (pb->num != num && pb->next != NULL) //循环找,要删除的结点
{
pf = pb;
pb = pb->next;
}
if (pb->num == num) //找到了一个结点的num和num相同
{
if (pb == *p_head) //要删除的是头结点
{
//让保存头结点的指针保存后一个结点的地址
*p_head = pb->next;
}
else
{
//前一个结点的指针域保存要删除的后一个结点的地址
pf->next = pb->next;
}
//释放空间
free(pb);
pb = NULL;
}
else //没有找到
{
printf("没有你要删除的结点\n");
}
}
//链表排序
void link_order(STU* head)
{
STU* pb, * pf, temp;
pf = head;
if (head == NULL)
{
printf("链表为空 不用排序\n");
return;
}
if (head->next == NULL)
{
printf("链表为空 不用排序\n");
return;
}
while (pf->next!=NULL) //以pf指向的结点为基准结点
{
pb = pf->next; //pb从基准元素的下一个元素开始
while (pb!=NULL)
{
if (pf->num > pb->num)
{
temp = *pb;
*pb = *pf;
*pf = temp;
temp.next = pb->next;
pb->next = pf->next;
pf->next = temp.next;
}
pb = pb->next;
}
pf = pf->next;
}
}
int main()
{
STU* head = NULL, * p_new = NULL, * pb = NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for (i = 0; i < num; i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
link_creat_head(&head, p_new); //将节点加入新的链表
}
link_print(head);
//link_free(&head);
link_order(head);
printf("排序后:\n");
link_print(head);
}
结果:
请输入链表初始个数:3
请输入学号,分数,名字:
6 98 l
请输入学号,分数,名字:
3 9 j
请输入学号,分数,名字:
9 87 h
num=6 score=98 name=l
num=3 score=9 name=j
num=9 score=87 name=h
排序后:
num=3 score=9 name=j
num=6 score=98 name=l
num=9 score=87 name=h
双向链表:一个结点由一个数据域,两个指针域(保存上一个节点的地址 保存下一个结点的地址)
1.双向链表的创建和遍历
第一步:创建一个结点作为头结点 将两个指针域都保存NULL
第二步:先找到链表中的最后一个结点 然后让最后一个结点的指针域保存新节点插入结点的地址,新插入结点的两个指针域,一个保存上一个节点地址 一个保存NULL
代码8:(有点问题)
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#pragma warning(disable:4996)
//定义结构体
typedef struct student
{
//数据域
int num;
int score;
char name[20];
//指针域
struct student* front;//保存上一个结点的地址
struct student* next; //保存下一个结点的地址
}STU;
void double_link_creat_head(STU** p_head, STU* p_new)
{
STU* p_mov = *p_head;
if (*p_head == NULL) //当第一次加入链表为空时,head执行p_new
{
*p_head = p_new;
p_new->next = NULL;
p_new->next = NULL;
}
else //第二次及以后加入链表
{
while (p_mov->next != NULL)
{
p_mov = p_mov->next; //找到原有链表的最后一个节点
}
p_mov->next = p_new;//将新申请的节点加入链表
p_new->front = p_mov;
p_new->next = NULL;
}
}
//链表的遍历
void double_link_print(STU* head)
{
STU* pb;
pb = head;
while (pb->next != NULL) //从头打印到尾
{
printf("num=%d score=%d name=%s \n", pb->num, pb->score, pb->name);
pb = pb->next;
}
printf("num=%d score=%d name=%s \n", pb->num, pb->score, pb->name);
printf("******************");
while (pb != NULL) //从尾打印到头
{
printf("num=%d score=%d name=%s \n", pb->num, pb->score, pb->name);
pb = pb->front;
}
}
//链表的释放
void link_free(STU** p_head)
{
//定义一个指针变量保存头指针的地址
STU* pb = *p_head;
while (*p_head != NULL)
{
//先保存p_head指向的结点的地址
pb = *p_head;
//p_head保存下一个结点的地址
*p_head = (*p_head)->next;
//释放结点并防止野指针
free(pb);
pb = NULL;
}
}
//链表结点的删除
void link_detele_num(STU** p_head, int num)
{
STU* pb, * pf;
pb = pf = *p_head;
if (*p_head == NULL) //链表为空 不用删
{
printf("链表为空,没有您要删的结点");
return;
}
while (pb->num != num && pb->next != NULL) //循环找,要删除的结点
{
pf = pb;
pb = pb->next;
}
if (pb->num == num) //找到了一个结点的num和num相同
{
if (pb == *p_head) //要删除的是头结点
{
//让保存头结点的指针保存后一个结点的地址
*p_head = pb->next;
}
else
{
//前一个结点的指针域保存要删除的后一个结点的地址
pf->next = pb->next;
}
//释放空间
free(pb);
pb = NULL;
}
else //没有找到
{
printf("没有你要删除的结点\n");
}
}
int main()
{
STU* head = NULL, * p_new = NULL, * pb = NULL;
int num, i;
printf("请输入链表初始个数:");
scanf("%d", &num);
for (i = 0; i < num; i++)
{
p_new = (STU*)malloc(sizeof(STU)); //申请一个新节点
printf("请输入学号,分数,名字:\n");
scanf("%d %d %s", &p_new->num, &p_new->score, p_new->name);
double_link_creat_head(&head, p_new); //将节点加入新的链表
}
double_link_print(head);
//link_free(&head);
}