不带头节点的链式存储实现链栈

1.先创建一个结构体类型,有数据域和指针域

复制代码
typedef struct LNode
{
	int data;
	struct LNode* next;
}LNode,*LinkStack;

2.以头节点为栈口进行操作进栈和出栈

头节点进栈

复制代码
int HeadPush(LinkStack* Ps, int elem)
{
	if ((*Ps) == NULL)
	{
		(*Ps) = (LNode*)malloc(sizeof(LNode));
		if ((*Ps) == NULL)
		{
			return 1;
		}
		(*Ps)->data = elem;
		(*Ps)->next = NULL;
		return 0;
	}
	LNode* s = (LNode*)malloc(sizeof(LNode));
	if (NULL ==s)
	{
		return 1;
	}
	s->data = elem;
	s->next = (*Ps);
	(*Ps) = s;
	return 0;
}

头节点出栈

复制代码
int HeadPop(LinkStack* Ps)//头节点出栈
{
	if ((*Ps) == NULL)
	{
		return 1;
	}
	LNode* p = (*Ps);
	(*Ps) = (*Ps)->next;
	free(p);
	return 0;
}
LNode* GetTail(LinkStack* Ps)
{
	LNode* p = (*Ps);
	if (NULL == p)
	{
		return p;
	}
	while (p->next)
	{
		p = p->next;
	}
	return p;
}

3.以尾节点为栈口进行操作进栈和出栈,因为是不带头节点的链栈,在处理尾节点的时候需要考虑到没有头节点,需要进行特殊处理

尾节点进栈

复制代码
int TailPush(LinkStack*Ps,int elem)
{
	if ((*Ps) == NULL)
	{
		(*Ps) = (LNode*)malloc(sizeof(LNode));
		if ((*Ps) == NULL)
		{
			return 1;
		}
		(*Ps)->next = NULL;
		(*Ps)->data = elem;
		return 0;
	}
	LNode* p = GetTail(Ps);//得到了最后一个非空的节点
	LNode* s = (LNode*)malloc(sizeof(LNode));
	if (NULL == s)
	{
		return 1;
	}
	s->data = elem;
	s->next = p->next;
	p->next = s;
	return 0;
}

尾节点出栈

复制代码
int TailPop(LinkStack* Ps)
{
	if ((*Ps) == NULL)//没有元素能够出栈
		return 1;
	//需要找到倒数第二个非空的节点
	if ((*Ps)->next == NULL)
	{
		LNode* temp = (*Ps);//只有一个元素
		(*Ps) = (*Ps)->next;
		free(temp);
		return 0;
	}
	//if ((*Ps)->next->next == NULL)
	//{
	//	LNode* temp = (*Ps)->next;
	//	(*Ps)->next = temp->next;
	//	free(temp);
	//	return 0;
	//}
	LNode* p = (*Ps);
	while (p->next->next)
	{
		p = p->next;
	}
	LNode* temp = p->next;
	p->next = temp->next;
	free(temp);
	return 0;
}

4.打印链栈

复制代码
void Display(LinkStack* Ps)
{
	LNode* p = (*Ps);
	while (p)
	{
		printf("%d->", p->data);
		p = p->next;
	}
	printf("NULL\n");
	
}

5.判断一个链栈是不是空的,如果不是空的返回top元素

以头节点进行操作的top节点

复制代码
void GetTop(LinkStack* Ps)//判断空栈和返回top元素
{
	if ((*Ps) == NULL)//说明是空表
	{
		return 1;
	}
	return (*Ps)->data;//返回栈顶元素
}

以尾节点进行操作的top节点

复制代码
LNode* GetTail(LinkStack* Ps)
{
	LNode* p = (*Ps);
	if (NULL == p)
	{
		return p;
	}
	while (p->next)
	{
		p = p->next;
	}
	return p;
}

6.全局代码,可以多测试几组,看是不是符合条件,主要是头节点和尾节点一定要考虑清楚

复制代码
typedef struct LNode
{
	int data;
	struct LNode* next;
}LNode,*LinkStack;

void InitStack(LinkStack* Ps)
{
	(*Ps) = NULL;//头节点为空指针
}
int HeadPush(LinkStack* Ps, int elem)
{
	if ((*Ps) == NULL)
	{
		(*Ps) = (LNode*)malloc(sizeof(LNode));
		if ((*Ps) == NULL)
		{
			return 1;
		}
		(*Ps)->data = elem;
		(*Ps)->next = NULL;
		return 0;
	}
	LNode* s = (LNode*)malloc(sizeof(LNode));
	if (NULL ==s)
	{
		return 1;
	}
	s->data = elem;
	s->next = (*Ps);
	(*Ps) = s;
	return 0;
}
int HeadPop(LinkStack* Ps)//头节点出栈
{
	if ((*Ps) == NULL)
	{
		return 1;
	}
	LNode* p = (*Ps);
	(*Ps) = (*Ps)->next;
	free(p);
	return 0;
}
LNode* GetTail(LinkStack* Ps)
{
	LNode* p = (*Ps);
	if (NULL == p)
	{
		return p;
	}
	while (p->next)
	{
		p = p->next;
	}
	return p;
}
int TailPush(LinkStack*Ps,int elem)
{
	if ((*Ps) == NULL)
	{
		(*Ps) = (LNode*)malloc(sizeof(LNode));
		if ((*Ps) == NULL)
		{
			return 1;
		}
		(*Ps)->next = NULL;
		(*Ps)->data = elem;
		return 0;
	}
	LNode* p = GetTail(Ps);//得到了最后一个非空的节点
	LNode* s = (LNode*)malloc(sizeof(LNode));
	if (NULL == s)
	{
		return 1;
	}
	s->data = elem;
	s->next = p->next;
	p->next = s;
	return 0;
}
int TailPop(LinkStack* Ps)
{
	if ((*Ps) == NULL)//没有元素能够出栈
		return 1;
	//需要找到倒数第二个非空的节点
	if ((*Ps)->next == NULL)
	{
		LNode* temp = (*Ps);//只有一个元素
		(*Ps) = (*Ps)->next;
		free(temp);
		return 0;
	}
	//if ((*Ps)->next->next == NULL)
	//{
	//	LNode* temp = (*Ps)->next;
	//	(*Ps)->next = temp->next;
	//	free(temp);
	//	return 0;
	//}
	LNode* p = (*Ps);
	while (p->next->next)
	{
		p = p->next;
	}
	LNode* temp = p->next;
	p->next = temp->next;
	free(temp);
	return 0;
}
void Display(LinkStack* Ps)
{
	LNode* p = (*Ps);
	while (p)
	{
		printf("%d->", p->data);
		p = p->next;
	}
	printf("NULL\n");
	
}
void GetTop(LinkStack* Ps)//判断空栈和返回top元素
{
	if ((*Ps) == NULL)//说明是空表
	{
		return 1;
	}
	return (*Ps)->data;//返回栈顶元素
}
int main()
{
	LinkStack S;
	InitStack(&S);
	//HeadPush(&S, 1);
	//HeadPush(&S, 2);
	//HeadPush(&S, 3);
	HeadPop(&S);
	TailPush(&S, 1);
	TailPush(&S, 2);
	TailPush(&S, 3);
	TailPush(&S, 4);
	TailPush(&S, 5);
	TailPop(&S);
	TailPop(&S);
	TailPop(&S);
	TailPop(&S);
	TailPop(&S);

	/*HeadPop(&S);
	HeadPop(&S);
	HeadPop(&S);*/

	Display(&S);
	return 0;
}
相关推荐
普贤莲花3 分钟前
【2026年第18周---写于20260501】---舍得
程序人生·算法·leetcode
2zcode3 分钟前
基于深度学习的口腔疾病图像识别系统(UI界面+改进算法+数据集+训练代码)
人工智能·深度学习·算法
Sarvartha12 分钟前
N 个字符串最长公共子序列(LCS)求解问题
数据结构·算法
一切皆是因缘际会12 分钟前
下一代 AI 架构:基于记忆演化与单向投影的安全智能系统
大数据·人工智能·深度学习·算法·安全·架构
falldeep19 分钟前
五分钟了解OpenClaw底层架构
人工智能·算法·机器学习·架构
m0_6294947319 分钟前
LeetCode 热题 100-----16.除了自身以外数组的乘积
数据结构·算法·leetcode
weixin_4462608525 分钟前
模型能力深度对决:GPT-4o、Claude 3.5和DeepSeek V系列模型的横向评测与未来趋势洞察
人工智能·算法·机器学习
迷途之人不知返42 分钟前
优先级队列:priority_queue
数据结构·c++
jieyucx1 小时前
Go 零基础数据结构:顺序表(像「排抽屉」一样学增删改查)
java·数据结构·golang
想唱rap1 小时前
应用层协议与序列化
linux·运维·服务器·网络·数据结构·c++·算法