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

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;
}
相关推荐
csdn_aspnet10 小时前
MATLAB 高效算法实战:数据分析与算法优化的效率秘诀
算法·matlab·数据分析
budingxiaomoli10 小时前
优选算法--链表
数据结构·算法·链表
漫随流水10 小时前
leetcode算法(637.二叉树的层平均值)
数据结构·算法·leetcode·二叉树
漫随流水11 小时前
leetcode算法(102.二叉树的层序遍历)
数据结构·算法·leetcode·二叉树
源代码•宸11 小时前
Leetcode—1339. 分裂二叉树的最大乘积【中等】
开发语言·后端·算法·leetcode·golang·dfs
leoufung11 小时前
LeetCode动态规划经典题:Unique Paths 网格路径计数详解
算法·leetcode·动态规划
李泽辉_11 小时前
深度学习算法学习(四):深度学习-最简单实现一个自行构造的找规律(机器学习)任务
深度学习·学习·算法
hz_zhangrl11 小时前
CCF-GESP 等级考试 2025年12月认证C++六级真题解析
c++·算法·青少年编程·程序设计·gesp·c++六级·gesp2025年12月
小沈同学呀11 小时前
基于时间片划分的提醒算法设计与实现
服务器·数据库·算法
千金裘换酒11 小时前
LeetCode 两数之和 Java
java·算法·leetcode