数据结构:栈和队列

目录

1.链表和顺序表优缺点

链表

优点:1.带头双向循环链表可在任意位置插入删除节点O(1)。2.按需申请释放空间。

缺点:1.不支持下标随机访问。2.CPU高速缓存命中率会更低。

顺序表

优点:1.尾插尾删效率较高。2.下标随机访问。3.CPU高速缓存命中率会更高。

缺点:1.前面部分插入删除数据,效率是O(N),需要挪动数据。2.空间不够,需要扩容。(扩容是需要付出代价的、一般还会伴随空间的浪费)。

2.栈

栈的概念及结构


实现栈的方式:数组栈

下面用一个动态数组来模拟栈(顺序表)。

c 复制代码
//需要用到的头文件
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<stdbool.h>
c 复制代码
typedef int STDataType;
typedef struct Stack
{
	STDataType* a;
	int top;//栈顶元素
	int capacity;//栈空间
}ST;//用顺序表来模拟栈
c 复制代码
//初始化栈
void STInit(ST* pst) 
{
	assert(pst);
	pst->a = NULL;
	//下面有两种方式来定义top
	//pst->top = -1;   //top 指向栈顶数据
	pst->top = 0;//top 指向栈顶数据的下一个位置

	pst->capacity = 0;
}
c 复制代码
//销毁栈
void STDestroy(ST* pst)
{
	asserta(pst);

	free(pst->a);
	pst->a = NULL;
	pst->top = pst->capacity = 0;
}
c 复制代码
//栈顶插入数据
void STPush(ST* pst, STDataType x)
{
	if (pst->top == pst->capacity)//扩容
	{
		int newCapacity = pst->capacity == 0 ? 4 : pst->capacity * 2;
		STDataType* tmp = (STDataType*)realloc(pst->a, newCapacity*sizeof(STDataType));
		if (tmp == NULL) {
			perror("realloc fail");
			return;
		}
		pst->a = tmp;
		pst->capacity = newCapacity;
	}

	pst->a[pst->top] = x;
	pst->top++;
}
c 复制代码
//栈顶出数据
void STPop(ST* pst)
{
	assert(pst);
	assert(!STEmpty(pst));//栈不为空再进行删除

	pst->top--;
}
c 复制代码
//获取栈顶元素
STDataType STTop(ST* pst)
{
	assert(pst);
	assert(!STEmpty(pst));//栈不为空再获取

	return pst->a[pst->top - 1];
}
c 复制代码
//探空
bool STEmpty(ST* pst)
{
	assert(pst);

	return pst->top == 0;
}
c 复制代码
//获取size
int STSize(ST* pst)
{
	assert(pst);

	return pst->top;
}

3.队列

队列的概念及结构

队列的实现:链式队列

c 复制代码
//需要用到的头文件
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<stdbool.h>
c 复制代码
typedef int QDataType;
typedef struct QueueNode
{
	struct QueueNode* next;
	QDataType data;
}QNode;//定义节点
typedef struct Queue
{
	QNode* phead;
	QNode* ptail;
	int size;
}Queue;
c 复制代码
//队列初始化
void QueueTnit(Queue* pq)
{
	assert(pq);

	pq->phead = NULL;
	pq->ptail = NULL;
	pq->size = 0;
}
c 复制代码
//队列销毁
void QueueDestroy(Queue* pq)
{
	assert(pq);

	QNode* cur = pq->phead;
	while (cur)
	{
		QNode* next = cur->next;
		free(cur);
		cur = next;
	}
	pq->phead = pq->ptail = NULL;
	pq->size = 0;
}
c 复制代码
//数据从队尾入队列
void QueuePush(Queue* pq, QDataType x)
{
	assert(pq);
	
	QNode* newNode = (QNode*)malloc(sizeof(QNode*));
	if (newNode == NULL)
	{
		perror("malloc fail\n");
		return;
	}
	newNode->data = x;
	newNode->next == NULL;
	if (pq->ptail == NULL)
	{
		assret(pq->phead == NULL);

		pq->phead = newNode;
	}
	else
	{
		pq->ptail->next = newNode;
		pq->ptail = newNode;
		pq->size++;
	}

}
c 复制代码
//数据从对头出队列
void QueuePop(Queue* pq)
{
	assert(pq);
	assert(!QueueEmpty(pq));

	//一个节点和多个节点
	if (pq->phead->next == NULL)//只有一个节点
	{
		free(pq->ptail);
		pq->phead = pq->ptail = NULL;
	}
	else//有多个节点
	{
		QNode* next = pq->phead->next;
		free(pq->phead);
		pq->phead = next;
	}
	pq->size--;
}
c 复制代码
//取对头数据
QDataType QueueFront(Queue* pq)
{
	assert(pq);
	assert(!QueueEmpty(pq));

	return pq->phead->data;
}
c 复制代码
//取队尾数据
QDataType QueueBack(Queue* pq)
{
	assert(pq);
	assert(!QueueEmpty(pq));

	return pq->ptail->data;
}
c 复制代码
//取队列数据个数
int QueueSize(Queue* pq)
{
	assert(pq);

	return pq->size;
}
c 复制代码
//判断队列是否为空
bool QueueEmpty(Queue* pq)
{
	assert(pq);

	return pq->phead == NULL
		&& pq->ptail == NULL;
}
相关推荐
kupeThinkPoem几秒前
跳表有哪些算法?
数据结构·算法
前端小L9 分钟前
图论专题(二十一):并查集的“工程应用”——拔线重连,修复「连通网络」
数据结构·算法·深度优先·图论·宽度优先
前端小L41 分钟前
图论专题(二十三):并查集的“数据清洗”——解决复杂的「账户合并」
数据结构·算法·安全·深度优先·图论
啊董dong1 小时前
课后作业-2025年11月23号作业
数据结构·c++·算法·深度优先·noi
dlz08362 小时前
从架构到数据结构,到同步逻辑,到 show run 流程优化
数据结构
jllws12 小时前
数据结构_字符和汉字的编码与查找
数据结构
学困昇2 小时前
C++11中的包装器
开发语言·数据结构·c++·c++11
weixin_4577600011 小时前
Python 数据结构
数据结构·windows·python
明洞日记12 小时前
【数据结构手册002】动态数组vector - 连续内存的艺术与科学
开发语言·数据结构·c++
fashion 道格12 小时前
数据结构实战:深入理解队列的链式结构与实现
c语言·数据结构