C++ 动态单链表基础

本文介绍了C++中的动态单链表基础知识,包括链表的定义、链表操作如创建、删除、插入、查找和输出,以及具体的操作流程。重点讲解了如何根据学号进行元素的插入和删除。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

这篇博客主要是带助教,好多学生提问了,就参考一篇博客稍微修改了一下

1.什么是链表

这里我只用一个简单的例子——线性单向链表为例,说明C语言是如何实现该结构的。
链表的元素是由结构体来实现struct Student *p。结构体中有一个成员是结构体指针struct Student *next,而这个结构体指针的类型和此结构体类型相同。除链表最后一个元素外,每一个结构体的指针都指向链表中下一个元素的结构体,最后一个元素的结构体指针为空(NULL)。保存链表时,只需要记录下链表的头指针,即链表中第一个结构体的地址即可。添加一个链表元素时,都需要单独申请一段内存;删除时则将其释放掉。在查找链表时,只需要顺着结构体指针的顺序一个一个往下查找,直到查找的结构体中的成员指针。

2.链表操作

链表的操作分为——创建表、插入元素、删除元素、清空表、查找表、打印表。其中插入/删除的元素可以是一个也可以说多个。链表从存储类型上来分可以分为静态链表和动态链表,静态链表是事先编写好的链表,占用的内存是静态存储区的内存,使用时不可以对其中的元素进行删减,只能查找;动态链表是按照程序要求生成的链表,存放于动态存储区,结构比较灵活,每一个元素都占据一部分存储空间,如果要删除元素,则释放该位置的内存;如果要添加元素,则申请一个结构体内存区的内存。

3.创建表 input

创建链表需要两个指针,一个作为先行指针(*p1),开辟内存并保存结构体的值;一个作为缓存指针(*p2),保留先行指针的所有值并且将它的next指向先行指针。构建链表时,先行指针赋一个值,后行指针保存一个值并且后行指针的next指向先行指针。赋值终止时,先行指针的next指向NULL,同时将先行指针赋值给后行指针,链表即构建完毕

4.删除元素 deletee

删除元素链表的第n个元素只需要将第n-1个元素的next指针指向第n+1个元素,再将第n个元素的内存释放即可,我这里是写的其中一个例子,根据关键字学号(int stdID)删除表中的某个元素,同时返回删除后的链表首地址(如果删的是第一个元素,则链表首地址会变)

5.插入元素 insertt

插入元素的原理是,假设要在第n个元素前插入一个元素。首先判断它是不是首元素,如果是,则修改头指针指向该元素,并将该元素的next指向原来的头指针。如果不是首元素,是第k个元素之前插入一个元素,则将第k-1个元素的next指针指向插入元素(或者子表)的地址(或者头指针),将插入元素的next指针(或尾指针)指向第k个元素。本示例代码是根据一个学号(主要关键字)插入一个元素(或者子表)的函数,返回链表的首地址(因为如果在第一个元素前面插入,可能改变链表的首地址)。

6.判断某个学生信息是否存在 isexist

在整个链表中通过学号ID查找,查找到了则返回,这里学号是唯一的没有相同的学号。

7.输出整个链表 output

从链表头开始遍历

8.操作流程

在这里插入图片描述
插入元素是指在某个元素之前插入,这里通过学号判断
在这里插入图片描述
在这里插入图片描述
续表是指在当前链表末尾输入元素,和前面的插入不一样
在这里插入图片描述
代码如下:

#include<iostream>
#include <malloc.h>
#include <stdbool.h>
#define LEN sizeof(struct Student)//定义结构体变量的大小为符号常量LEN
using namespace std;

struct Student{
	int ID;//学号
	string name;//姓名
	float height;//身高
	float weight;//体重
	float BMI;//BMI指数,录入时不需要计算
	struct Student *next;//指向下一个结构体
};
struct Student *input();//输入函数
void output(struct Student * head);//输出函数
struct Student *deletee(struct Student *head,int stdID);//删除一个元素,返回删除后表的头指针
struct Student *insertt(struct Student *head,int stdID,struct Student *insertstd);//返回插入元素(子表)后的头指针
int append(struct Student *head);//插入一个链表,从input函数输入
struct Student *isexist(struct Student *head,int stdID);

int main()
{
	struct Student *present;//当前链表的头指针
	int choice;
	bool next;
	int stdID;
	/*
	1:插入一个元素
	2:删除一个元素
	3:续表
	4:查找表
	*/
	printf("**********动态链表实验**********\n初始化一个链表:\n");
	present=input();//当前的链表指针
	do{
		printf("请选择:\n|1:插入元素(子表)\n|2:删除元素\n|3:续表\n|4:查找表\n");
		scanf("%d",&choice);
		switch(choice)
		{
			case 1:
				printf("请输入插入地点的后一个同学的学号: ");
				scanf("%d",&stdID);
				if(isexist(present,stdID)==NULL)
				{
					printf("该学生不存在!\n");
					break;//退出switch语句
				}
				present=insertt(present,stdID,input());
				printf("插入元素后的链表为:\n");
				output(present);
				break;
			case 2:
				printf("请输入删除元素的学号:  ");
				scanf("%d",&stdID);
				if(isexist(present,stdID)==NULL)
				{
					printf("该学生不存在!\n");
					break;//退出switch语句
				}
				present=deletee(present,stdID);
				printf("删除后的链表为:\n");
				output(present);
				break;
			case 3:
				append(present);
				printf("续表后的链表为:\n");
				output(present);
				break;
			case 4:
				printf("当前链表为:\n");
				output(present);
				break;
		}
		printf("是否继续(Yes:1,No:0):  ");
		scanf("%d",&next);
		fflush(stdin);
	}while(next);


	return 0;
}


struct Student * input()
{
	struct Student *p1,*p2,*head=NULL;
	printf("************************动态链表实验***********************\n【输入动态链表】\n");
	printf("请依次输入学号	姓名  身高(cm)  体重(kg)(用空格间隔,学号输入0结束):\n");
	p2=p1=(struct Student *)malloc(LEN);//开辟内存
	scanf("%d %s %f %f",&p1->ID,&p1->name,&p1->height,&p1->weight);
	if(p1->ID==0)return(head);
	else head=p1;
	while(p1->ID!=0)
	{
		p2->next=p1;
		p2=p1;
		p2->BMI=(float)p2->weight/(p2->height/100)/(p2->height/100);//求BMI指数
		p1=(struct Student *)malloc(LEN);//开辟内存
		scanf("%d",&p1->ID);
        if(p1->ID)
            scanf("%s %f %f",&p1->name,&p1->height,&p1->weight);
	}
	p2->next=NULL;
	return(head);//返回链表头指针
}

void output(struct Student *head)
{
	struct Student *p;
	int num=1;
	p=head;//将头指针地址传给p
	printf("【输出动态链表】\n");
	printf("|学号\t\t|姓名\t|身高\t|体重\t|BMI\n");
	while(p!=NULL)
	{
		printf("%3d|%08d\t|%s\t|%5.2f\t|%5.2f\t|%lf\n",num++,p->ID,p->name,p->height,p->weight,p->BMI);
		p=p->next;
	}
}

struct Student *deletee(struct Student *head,int stdID)
{
	struct Student *p1,*p2;
	if(head->ID==stdID)
	{
		p1=head->next;
		free(head);
		return p1;
	}//如果删除的是第一个元素,比较特殊,需要修改头指针,其余不动
	//剩余几种情况都是修改next结构体指针
	for(p1=head;p1!=NULL;p2=p1,p1=p1->next)//p1指针和p2指针同时查找,p1指向当前的学生,p2保指向了上一个学生
	{
		if(p1->ID==stdID)
		{
		 	p2->next=p1->next;//假设找到了需要删除的学生的学号,则让它上一个学生的指针指向跳过他的下一个学生
		 	free(p1);
		 	return head;
		}
	}
	return NULL;//返回NULL代表没找到
}

struct Student *insertt(struct Student *head,int stdID,struct Student *insertstd)
{
	struct Student *p1,*p2,*p;
	for(p=insertstd;p->next!=NULL;p=p->next);//找到insert链表的最后一个元素
	if(head->ID==stdID)
	{
		p->next=head;
		return insertstd;
	}

	for(p1=head;p1!=NULL;p2=p1,p1=p1->next)
	{
		if(p1->ID==stdID)
		{
			p2->next=insertstd;
			p->next=p1;
			return head;
		}
	}
	return NULL;
}

int append(struct Student *head)//插入一个链表,从input函数输入
{
	struct Student *p;
	for(p=head;p->next!=NULL;p=p->next);//找到head链表的最后一个元素
	p->next=input();//从input输入需要添加的元素,可以是1个或者多个
}

struct Student *isexist(struct Student *head,int stdID)
{
	struct Student *p;
	for(p=head;p!=NULL;p=p->next)
	{
		if(p->ID==stdID)
		{
			return p;
		}
	}
	return NULL;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值