栈和队列刷题 Leetcode.225/232/20【C语言实现】

简介: 栈和队列刷题 Leetcode.225/232/20【C语言实现】

声明:由于C的局限性,以下OJ题所用到的接口(如Init、Pop、Push等)都需要自己实现,详情请看C语言实现栈和队列

1. 有效的括号

思路:利用栈的结构特性:先进后出。

假设输入()[]{}

  1. 将所有的左括号压入栈中(([{)
  2. 剩下的右括号与离它最近的左括号匹配。如何匹配?——出栈
  • 函数调用的接口是自己实现的,唯一需要改动的是数据类型typedef char STDataType; 此处体现了类型重命名的好处。
  • 匹配前注意判空
  • 注意字符串的迭代
bool isValid(char * s){
    struct Stack st;
    StackInit(&st);
    while(*s)
    {
        if(*s == '[' || *s == '(' || *s == '{')
        {
            StackPush(&st, *s);//压栈
            s++;//字符串迭代
        }
        else
        {
            if(StackEmpty(&st))//前面无括号
            {
                StackDestroy(&st);
                return false;
            }
            else
            {
                char top = StackTop(&st);//得到栈顶元素
                if((top == '[' && *s != ']')//不匹配的情况
                || (top == '(' && *s != ')')
                || (top == '{' && *s != '}'))
                {
                    StackDestroy(&st);
                    return false;
                }
                else//剩下的是匹配的情况
                {
                    StackPop(&st);//出栈,下次循环开始得到下一个元素
                    s++;//字符串迭代
                }
            }
        }
    }
    //如果栈中都匹配完了,就不会剩下括号
    bool ret = StackEmpty(&st);//改接口判断栈是否为空
    StackDestroy(&st);
    return ret;
}

2. 用队列实现栈

思路:用队列实现栈,实际上就是用先进先出实现先进后出的效果。可以用两个队列来实现。

  • 入数据:往非空的队列入数据
  • 出数据:将最后一个元素之前的所有元素放到另一个空队列中,然后把剩下的一个返回后删除

2.1 结构的定义

用队列实现栈,MyStack这个结构就要用两个队列作为成员

typedef struct {
    Queue q1;
    Queue q2;
} MyStack;

2.2 队列的初始化和销毁

要先为结构体(队列)开辟空间,然后再初始化两个队列

MyStack* myStackCreate() {
    MyStack* pst = (MyStack*)malloc(sizeof(MyStack));
    QueueInit(&pst->q1);
    QueueInit(&pst->q2);
    return pst;
}

注意free的顺序,从结构体内到外

void myStackFree(MyStack* obj) {
    QueueDestroy(&obj->q1);
    QueueDestroy(&obj->q2);
    free(obj);
}

2.3 模拟入栈

用入队模拟入栈。

  • 注意入队的是非空队列
void myStackPush(MyStack* obj, int x) {
    if(!QueueEmpty(&obj->q1))
    {
        QueuePush(&obj->q1, x);
    }
    else
    {
        QueuePush(&obj->q2, x);
    }
}

2.4 模拟出栈

假设入栈的是1、2、3、4,出栈第一个数据是4

首先假设空的是q1,非空是q2,如果事实相反,则用条件判断反置

然后将非空队列的前size-1个(要调用size接口)放到空的队列,边放边删除(这样才能放下一个)。

放完以后把剩下的那一个没放的删掉。但是这个数据(4)就是要出栈的数据,所以要用一个变量保存它并 返回,这样就达到了出栈的效果。

int myStackPop(MyStack* obj) {
    //假设
    Queue* pEmpty = &obj->q1;
    Queue* pNonEmpty = &obj->q2;
    if(!QueueEmpty(&obj->q1))//如果假设不成立,反置赋值语句
    {
        pEmpty = &obj->q2;
        pNonEmpty = &obj->q1;
    }
    while(QueueSize(pNonEmpty) > 1)//留下最后一个要出栈的数据
    {
        //将数据放到空队列中
        QueuePush(pEmpty, QueueFront(pNonEmpty));
        //边放边删除非空队列的数据
        QueuePop(pNonEmpty);
    }
    //在删除最后一个要出栈的数据之前保存并返回它
    int front = QueueFront(pNonEmpty);
    QueuePop(pNonEmpty);
    return front;
}

2.5 模拟获取栈顶数据

队列一定是一个为空,一个非空,将非空队列中的队尾数据返回

例如1、2、3、4,栈顶数据为4,它处于队尾

int myStackTop(MyStack* obj) {
    if(!QueueEmpty(&obj->q1))
    {
        return QueueBack(&obj->q1);
    }
    else
    {
        return QueueBack(&obj->q2);
    }
}

2.6 判空

因为模拟的栈中是两个队列,所以两个队列都要判断

bool myStackEmpty(MyStack* obj) {
    return QueueEmpty(&obj->q1) && QueueEmpty(&obj->q2);
}

3. 用栈实现队列

思路:

思路和上一题类似,同样是用两个栈(先进后出)模拟实现队列(先进先出)。不同的是,两个栈各自的功能不同,一个只出栈,一个只入栈。

ps:这里的思路有点像反转字符串。每次出栈都是调换一次顺序,出两次栈就抵消了。

3.1 结构的定义

用栈模拟实现队列,需要连续出栈两次,所以要定义两个栈

typedef struct {
    Stack pushSt;
    Stack popSt;
} MyQueue;

3.2 栈的初始化和销毁

首先要为队列开辟空间,然后初始化栈

MyQueue* myQueueCreate() {
    MyQueue* q = (MyQueue*)malloc(sizeof(MyQueue));
    StackInit(&q->pushSt);
    StackInit(&q->popSt);
    return q;
}

销毁注意顺序

void myQueueFree(MyQueue* obj) {
    StackDestroy(&obj->pushSt);
    StackDestroy(&obj->popSt);
    free(obj);
}

3.3 判空

判断队列是否为空,即判断两个栈是否为空

bool myQueueEmpty(MyQueue* obj) {
    return StackEmpty(&obj->pushSt) && StackEmpty(&obj->popSt);
}

3.4 压栈

直接压入即可

void myQueuePush(MyQueue* obj, int x) {
    StackPush(&obj->pushSt, x);
}

3.5 获取栈顶元素

  • 要把非空的pushST的所有元素放到空的popST中,注意判断条件
  • 在把pushST元素放入popST中后,记得删除该元素
  • 最后才是返回popST的栈顶元素
int myQueuePeek(MyQueue* obj) {
    if(StackEmpty(&obj->popSt))
    {
        while(!StackEmpty(&obj->pushSt))
        {
            StackPush(&obj->popSt, StackTop(&obj->pushSt));
            StackPop(&obj->pushSt);
        }
    }
    return StackTop(&obj->popSt);
}

3.6 出栈

边出栈,边删除,用top保存popST的栈顶元素并返回

int myQueuePop(MyQueue* obj) {
    int top = myQueuePeek(obj);
    StackPop(&obj->popSt);
    return top;
}

4. 设计循环队列

详见循环队列


日志

4/23/2022
目录
相关文章
|
2月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
262 9
|
4月前
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
78 5
|
2月前
|
机器学习/深度学习 人工智能 自然语言处理
280页PDF,全方位评估OpenAI o1,Leetcode刷题准确率竟这么高
【10月更文挑战第24天】近年来,OpenAI的o1模型在大型语言模型(LLMs)中脱颖而出,展现出卓越的推理能力和知识整合能力。基于Transformer架构,o1模型采用了链式思维和强化学习等先进技术,显著提升了其在编程竞赛、医学影像报告生成、数学问题解决、自然语言推理和芯片设计等领域的表现。本文将全面评估o1模型的性能及其对AI研究和应用的潜在影响。
59 1
|
4月前
|
数据采集 负载均衡 安全
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
本文提供了多个多线程编程问题的解决方案,包括设计有限阻塞队列、多线程网页爬虫、红绿灯路口等,每个问题都给出了至少一种实现方法,涵盖了互斥锁、条件变量、信号量等线程同步机制的使用。
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
|
3月前
|
C语言
数组栈的实现(C语言描述)
本文介绍了如何在C语言中使用数组来实现栈的数据结构,包括栈的创建、入栈、出栈、获取栈顶元素、检查栈是否为空、获取栈的大小以及销毁栈等操作,并提供了相应的函数实现。
47 1
|
3月前
|
机器学习/深度学习 编译器 C语言
C语言刷题(中)(保姆式详解)
C语言刷题(中)(保姆式详解)
23 0
|
3月前
【LeetCode 24】225.用队列实现栈
【LeetCode 24】225.用队列实现栈
19 0
|
3月前
|
算法
【LeetCode 23】232.用栈实现队列
【LeetCode 23】232.用栈实现队列
27 0
|
4月前
|
存储 人工智能 C语言
数据结构基础详解(C语言): 栈的括号匹配(实战)与栈的表达式求值&&特殊矩阵的压缩存储
本文首先介绍了栈的应用之一——括号匹配,利用栈的特性实现左右括号的匹配检测。接着详细描述了南京理工大学的一道编程题,要求判断输入字符串中的括号是否正确匹配,并给出了完整的代码示例。此外,还探讨了栈在表达式求值中的应用,包括中缀、后缀和前缀表达式的转换与计算方法。最后,文章介绍了矩阵的压缩存储技术,涵盖对称矩阵、三角矩阵及稀疏矩阵的不同压缩存储策略,提高存储效率。
513 8