当前位置: 首页 > article >正文

9.24 数据结构-栈、队列总结

一、栈

1.1 概念

1、操作受限的线性表,只能在一端插入删除。

2、先进后出或者后进先出的线性表。

1.2 顺序栈

1.2.1 顺序栈的概念

1、借助数组来存储的栈

1.2.2 顺序栈的组成

1、用来存储栈的数组

2、记录下标的top

1.2.3 顺序栈结构体类型
typedef struct
{
    int data[MAX];//存储栈元素的数组
    int top;//记录栈顶元素的下标
}stack,*Pstack;
1.2.4 顺序栈的相关操作(功能函数)

1> 创建栈 
Pstack create()
{
    Pstack p = malloc(sizeof(stack));
    if(NULL==p)
    {
        printf("创建栈失败\n");
        return NULL;
    }
    p->top = -1;
    return p;//返回栈地址
}
 2> 入栈

1、判断栈满

2、入栈先加后压

int push(Pstack Z,int e)
{
    if(Z->top==MAX-1)//判断栈满
    {
        printf("栈满无法入栈\n");
        return -1;
    }
    //入栈先加后压
    Z->top++;
    Z->data[Z->top] = e;
    printf("入栈成功\n");
    return 0;
}
 3> 遍历栈

3.1从栈底到栈顶遍历

int output1(Pstack Z)
{
    if(Z->top==-1)
    {
        printf("栈空,无法输出\n");
        return -1;
    }

    int i;
    for(i  =0;i<=Z->top;i++)
    {
        printf("%d\t",Z->data[i]);
    }
    printf("\n");
    return 0;
}

3.2从栈顶到栈底遍历

int output2(Pstack Z)
{
    if(Z->top==-1)
    {
        printf("栈空,无法输出\n");
        return -1;
    }

    int i;
    for(i  =Z->top;i>=0;i--)
    {
        printf("%d\t",Z->data[i]);
    }
    return 0;
}
4> 出栈

1、判断栈空

2、先弹后减

int pop(Pstack Z)
{
    if(Z->top==-1)//判断栈空
    {
        printf("栈空无法出栈\n");
        return -1;
    }
    //出栈先弹后减
    printf("%d\t",Z->data[Z->top]);
    Z->top--;
    return 0;
}
5> 获取栈顶元素
int huoqu(Pstack Z)
{
    if(Z->top==-1)
    {
        printf("获取失败\n");
        return -1;
    }
    return Z->data[Z->top];
}
6>求栈的大小
int sizestack(Pstack Z)
{
    return Z->top+1;
}
7>销毁栈
int destroy_stack(Pstack Z)
{
    free(Z);
    Z=NULL;
    printf("销毁成功\n");
    return 0;
}

1.3链式栈

1>创建链式栈 
typedef struct node//正常节点结构体
{
    int data;
    struct node *next;
}xxx,*Pxxx;
typedef struct//栈顶指针的结构体
{
    int len;
    Pxxx top;
}stack,*Pstack;
Pstack create()//创建栈顶指针节点
{
    Pstack p = malloc(sizeof(stack));
    if(p==NULL)
    {
        printf("申请节点失败\n");
        return NULL;
    }
    p->len=0;
    p->top=NULL;
    return p;
}
2>入栈

int push(Pstack S,int e)//入栈
{
    if(S==NULL)
    {
        printf("入栈失败\n");
        return -1;
    }
    Pxxx p = malloc(sizeof(xxx));
    p->data = e;

    p->next = S->top;
    S->top = p;

    S->len++;
    printf("入栈成功\n");
    return 0;
}
3> 出栈

int pop(Pstack S)//出栈
{
    if(S->top==NULL||S==NULL)
    {
        printf("栈空出栈失败\n");
        return -1;
    }

    Pxxx Q = S->top;
    printf("出栈元素是:%d\n",Q->data);
    S->top = Q->next;
    free(Q);
    Q = NULL;
    S->len--;
    return 0;
}
 4>遍历
int ouput_stack(Pstack S)//输出
{
    if(S==NULL||S->top==NULL)
    {
        printf("栈不存在或者空\n");
        return -1;
    }
    Pxxx t = S->top;
    while(t!=NULL)
    {
        printf("%d\t",t->data);
        t = t->next;
    }
    return 0;
}

二、队列

2.1 概念

1、操作受限的线性表,只能队尾入队,队头出队。

2、先进先出,后进后出的线性表。

2.2 顺序队列

2.2.1 概念

1、借助数组来存储的队列。

2.2.2 组成

1、存储队列元素的数组

2、队头下标

3、队尾下标

2.2.3 假溢满现象:

 2.3 循环顺序队列

出队,入队,队长推导:

2.3.1 引入目的

循环队列可以解决顺序队列的假溢出现象。

2.3.2 队列结构体类型
#define MAX 5
typedef struct//循环队列结构体
{
    int data[MAX];
    int rear;
    int front;
}queue,*Pqueue;
2.3.3 相关操作(功能函数) 
1> 创建循环队列
Pqueue create()//创建循环队列
{
    Pqueue p = malloc(sizeof(queue));
    if(NULL==p)
    {
        printf("创建循环队列失败\n");
        return NULL;
    }
    p->rear = 0;
    p->front = 0;
    return p;
}
2>入队
int enqueue(Pqueue D,int e)
{
    if((D->rear+1)%MAX==D->front)
    {
        printf("队满无法入队\n");
        return -1;
    }
    D->data[D->rear]  =e;//入队
    D->rear = (D->rear+1)%MAX;//队尾下标+1
    printf("入队成功\n");
    return 0;
}
3>遍历
int output(PQueue D){
	//输出队头到队尾信息//可以分情况讨论也可以直接(D->rear-D->front+MAX)%MAX队列元素长度做判断条件
	int i;
	if(D->rear>D->front){
		for(i=D->front;i<D->rear;i++){
			printf("%d\t",D->data[i]);
		}
	}else{
		//front-->MAX-1
		for(i=D->front;i<=MAX-1;i++){
			printf("%d\t",D->data[i]);
		}
		//0->rear
		for(i=0;i<D->rear;i++){//因为D->rear指向队列空白下标,所以是<不是<=
			printf("%d\t",D->data[i]);
		}
	}
}

 4>出队

int outqueue(Pqueue D)
{
    if(D->rear==D->front)
    {
        printf("队列为空无法出队\n");
        return -1;
    }
    printf("出队元素:%d\n",D->data[D->front]);//出队
    D->front = (D->front+1)%MAX;//队头下标+1
    return 0;
}
5> 求队列长度 

 6>释放队列
int destroy(Pqueue D)
{
    if(D==NULL)
    {
        printf("销毁失败\n");
        return -1;
    }
    free(D);
    D = NULL;
    return 0;
}

2.4链式队列

 2.4.1 链式队列结构体类型
typedef struct node
{
    int data;
    struct node *next;        
}xxx,*Pxxx;
typedef struct 
{
    int len;
    Pxxx rear;
    Pxxx front;
}queue,*Pqueue;
 2.4.2 链式队列的相关操作(功能函数)
1>创建
Pqueue create()//创建队列头结点
{
    Pqueue p = malloc(sizeof(queue));
    if(p==NULL)
    {
        printf("申请头结点失败\n");
        return NULL;
    }
    p->len=0;
    p->rear = NULL;
    p->front = NULL;
    return p;
}
2>入队

1、没有节点时

p->next = NULL;

D->rear=p

D->front = p;

2、有节点时

p->next = NULL;

D->rear->next = p;

D->rear = p;

int enqueue(Pqueue D,int e)
{
    if(D==NULL)
    {
        printf("入队失败\n");
        return -1;
    }
    Pxxx p = malloc(sizeof(xxx));
    p->data = e;
    if(D->rear==NULL)//没有节点
    {
        D->front = p;
        D->rear = p;
        p->next = NULL;
    }
    else//有节点
    {
        D->rear->next = p;
        D->rear = p;
        p->next  =NULL;
    }
    D->len++;
    return 0;
3>出队

当有多个节点时:

当有1个节点时 

 

int outqueue(Pqueue D)
{
    if(D==NULL||D->rear==NULL)
    {
        printf("队列为空\n");
        return -1;
    }
    Pxxx Q;
    if(D->rear!=D->front)//不只一个节点
    {
        Q = D->front;//保留要出队的节点
        printf("出队元素:%d\n",Q->data);
        D->front = Q->next;
    }
    else//只有一个节点
    {
        Q = D->front;//保留要出队的节点
        printf("出队元素:%d\n",Q->data);
        D->front = D->rear= NULL;

    }
        free(Q);
        Q = NULL;
        D->len--;
        return 0;

}
4> 遍历 
int output(Pqueue D)
{
    if(D==NULL||D->rear==NULL)
    {
        printf("没有节点输出失败\n");
        return -1;
    }
    int i;
    Pxxx t  = D->front;
    for(i = 0;i<D->len;i++)
    {
        printf("%d\t",t->data);
        t = t->next;
    }
    return 0;
}
 5> 销毁队列
int destroy(Pqueue D)
{
    Pxxx t;
    while(D->front!=NULL)
    {
        t = D->front;
        D->front = D->front->next;
        free(t);
    }
    free(D);
    D=NULL;
    printf("销毁成功\n");
}

 


http://www.kler.cn/a/328938.html

相关文章:

  • 高并发处理 --- 超卖问题+一人一单解决方案
  • ray.rllib 入门实践-2:配置算法
  • 豆瓣Top250电影的数据采集与可视化分析(scrapy+mysql+matplotlib)
  • 6. 马科维茨资产组合模型+政策意图AI金融智能体(DeepSeek-V3)增强方案(理论+Python实战)
  • Hive之加载csv格式数据到hive
  • SpringBoot使用MockMVC通过http请求controller控制器调用测试
  • 蓝桥杯—STM32G431RBT6(IIC通信--EEPROM(AT24C02)存储器进行通信)
  • 【深度学习】05-Rnn循环神经网络-04- RNN中的权重和偏置共享指的是什么?/ 为什么要共享/以及怎么进行记忆传递的?
  • Python | Leetcode Python题解之第441题排列硬币
  • Springboot结合RabbitMQ
  • 经典文献阅读之--Stereo-NEC(全新双目VIO初始化)
  • web前端-CSS引入方式
  • Vue3 工具函数(总结)
  • Python和QT哪个更适合嵌入式方向的上位机开发?
  • 【计算机毕业设计】springboot就业信息管理系统
  • Java中的HTTP请求:使用Apache HttpClient
  • python程序操作Windows系统中的软件如word等(是否可以成功操作待验证)
  • 计算机网络实验3——基于TCP的多线程Web Server服务器的实现
  • vue页面保持在div的底部(适用于聊天界面等需要显示最新信息的场景)
  • R包:ggheatmapper热图
  • Postgresql源码(136)syscache/relcache 缓存及失效机制
  • 【数据结构】环形队列(循环队列)学习笔记总结
  • 技术人生-电脑突然卡顿怎么办
  • 滚雪球学Oracle[3.4讲]:事务控制与锁管理
  • Vite:为什么选 Vite
  • 22.4k star,好用、强大的链路监控软件,skywalking