c语言之链表

定义:链表是一种物理储存上非连续,数据元素的逻辑顺序通过链表中的指针链接次序,实现的一种现行储存结构。
特点:
链表由一系列节点(链表中每一个元素称为节点)组成,节点在运行时动态生成(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);

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值