单链表练习题-向有序单链表中插入元素并保持有序(C语言实现)

单链表练习题-向有序单链表中插入元素并保持有序(C语言实现)

一、题目

​ 如图,原本链表就有序:
在这里插入图片描述
假设现在要插入元素15,我们希望它插入在10和16之间。
假设现在要插入元素5,我们希望它插入在7的前面。
假设现在要插入元素22,我们希望它插入在20后面。

二、思路

​ 由于原来的链表已经有序,那么我们只需要一个指针就可以了。用指针p遍历链表,比较p当前所指元素和p后继的元素,如果要插入的元素大于等于p所指元素,而且小于等于p的后继元素,这时就可以在p之后插入该元素,否则继续遍历下去,直到链表末尾。

三、动手实现

bool Insert_Order_List(LinkList *L,ElemType e)//有序插入
{
	LinkList *p = L;
	
	while(p->next!=NULL)
	{
		if((e>=p->data)&&(e<=p->next->data))
		{
			//申请一个新的结点
			LinkList *s = (LinkList *)malloc(sizeof(LinkList));
			s->data = e;//赋值 
			
			//修改指针  将结点s插入到结点p之后 
			s->next = p->next;//s指针指向
			p->next = s;
			break;//插入之后跳出while循环	
		}
		else
			p = p->next;
	}
	return true;
} 

四、全部代码

#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h> //根据C99标准,C语言使用bool类型需要添加这个头文件

typedef int ElemType;

typedef struct LinkNode
{
	ElemType data;
	struct LinkNode *next;
}LinkList;

//------------ 函数声明 ----------//
void MainMenu();
bool InitLinkList(LinkList *L);//初始化 
bool InsertLinkList(LinkList *L,ElemType e);//插入
void PrintAll(LinkList *L);//输出所有元素 
bool Insert_Order_List(LinkList *L,ElemType e);//有序插入 
//------------ 函数声明 ----------//

int main()
{
	LinkList L;
	
	int ch; 
	ElemType element;
	
	if(InitLinkList(&L))
		printf("初始化成功!\n");
	else
		printf("初始化失败!\n");
	
	while(1)
	{
		MainMenu(); 
		printf("请输入您要执行的操作:");
		scanf("%d",&ch);
		
		switch(ch)
		{
			case 0:		printf("感谢使用,已退出!");	exit(0);	break;
			case 1:		printf("请输入您要插入的元素:\n");
						scanf("%d",&element); 
						if(InsertLinkList(&L,element))
							printf("插入成功!\n");
						else
							printf("插入失败!\n");
						break;
			case 2:		PrintAll(&L);
						break;		
						
			case 3:		printf("请输入您要插入的元素:\n");
						scanf("%d",&element); 
						if(Insert_Order_List(&L,element))
							printf("有序插入成功!\n");
						else
							printf("有序插入失败!\n");
						break;
			default:	printf("您输入的操作指令有误!请重新输入!");
		}
	}
	
	return 0;
}

//主菜单,显示 
void MainMenu()
{
	printf("\n\n\n");
	printf("\t      **** 向有序单链表中插入元素并保持有序 ****\n\n"); 
	printf("\t      -------	0.退出 \n\n");
	printf("\t      -------	1.插入元素\n\n");
	printf("\t      -------	2.输出所有元素\n\n");
	printf("\t      -------	3.插入单链表并保持有序\n\n");
	printf("\t      *************************************\n");
}


//初始化单链表(带头结点) 
bool InitLinkList(LinkList *L)
{
	//先申请一个头结点
	LinkList *head = (LinkList *)malloc(sizeof(LinkList));

	L->next = head;
	head->next = NULL;//头结点之后一开始还没元素
	return true;
} 

//插入
bool InsertLinkList(LinkList *L,ElemType e)
{
	//头插法
	LinkList *p = L;//
	
	//申请一个新的结点
	LinkList *s = (LinkList *)malloc(sizeof(LinkList));
	
	s->data = e;//赋值 
	
	//修改指针  将结点s插入到结点p之后 
	s->next = p->next;//s指针指向
	p->next = s;

	return true;
} 

//打印所有元素 
void PrintAll(LinkList *L)
{
	LinkList *q;
	q = L->next;

	//打印出所有元素 
	while(q->next!=NULL)
	{
		printf("%d ",q->data);
		q = q->next;
	}
}

bool Insert_Order_List(LinkList *L,ElemType e)//有序插入
{
	LinkList *p = L;
	
	while(p->next!=NULL)
	{
		if((e>=p->data)&&(e<=p->next->data))
		{
			//申请一个新的结点
			LinkList *s = (LinkList *)malloc(sizeof(LinkList));
			
			s->data = e;//赋值 
			
			//修改指针  将结点s插入到结点p之后 
			s->next = p->next;//s指针指向
			p->next = s;
			break;//插入之后跳出while循环	
		}
		else
			p = p->next;
	}
	return true;
} 

五、测试

1.先插入一些元素并输出
在这里插入图片描述
2.插入5,并查看结果:
在这里插入图片描述
在这里插入图片描述
3.插入20,并查看结果:
在这里插入图片描述
在这里插入图片描述
4.插入15,并查看结果:
在这里插入图片描述

在这里插入图片描述
可以看到无论是边界还是中间都已成功有序插入。

C语言,你可以通过遍历有序单链表并比较每个节点的实现这个功能。这里是一个简单的示例代码,它假设`struct Node`是一个包含`data`和`next`指针的链表节点结构体: ```c #include <stdio.h> #include <stdlib.h> // 定义链表节点结构 typedef struct Node { int data; struct Node* next; } Node; // 插入保持链表有序的辅助函数 Node* replace_e(Node* head, int e, int e1) { if (head == NULL || head->data > e) { // 如果链表为空或者头节点大于e,直接插入新的e1 head = (Node*)malloc(sizeof(Node)); head->data = e1; head->next = head; } else { Node* current = head; while (current->data != e && current->next != NULL && current->next->data <= e) { current = current->next; } // 找到第一个等于e的节点,将其数据替换为e1 if (current->data == e) { current->data = e1; } } return head; } // 主函数 int main() { // 假设你已经有一个已排序的链表,例如:1->2->3->4->5->NULL Node* list = create_sorted_list(); // 这里你需要自定义创建链表的函数 // 要替换的e和新e1 int e = 3; int e1 = 7; // 替换操作 list = replace_e(list, e, e1); // 输出处理后的链表(假设链表打印函数print_list也已定义) print_list(list); // 清理内存 free_list(list); // 这里需要自定义释放链表节点的函数 return 0; } ``` 注意,这只是一个基本的示例,实际项目你可能需要处理更多边界情况,并且上述代码没有包括链表的创建、删除和释放等其他功能,你需要根据实际情况完善。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值