https://leetcode-cn.com/problems/valid-parentheses/
遇到左括号就入栈,遇到右括号就取出栈顶元素与之相比,如果不配对就返回假,否则继续。需要判断一些特殊情况,比如全是左括号。
本代码是用C语言写的,需要在前面添加栈的实现相关函数。
bool isValid(char * s)
{
ST st;
StackInit(&st);
while(*s)
{
//当s没有走到末尾
if(*s=='('||*s=='['||*s=='{')
{
//左括号就入栈
StackPush(&st,*s);
}
else
{
//右括号就出栈,与栈顶元素比较
if(StackEmpty(&st))
{
//如果栈空就不符合
StackDestroy(&st);
return false;
}
char top=StackTop(&st);
StackPop(&st);
//查看是否满足条件
if((*s==')'&&top=='(') ||
(*s==']'&&top=='[')||
(*s=='}'&&top=='{'))
{
//continue;//匹配正确就继续
}
else
{
StackDestroy(&st);
return false;
}
}
s++;
}
//全部检查一遍后,栈不为空,说明还有未配对的左括号
if(!StackEmpty(&st))
{
//栈不为空
StackDestroy(&st);
return false;
}
else
{
StackDestroy(&st);
return true;
}
}
https://leetcode-cn.com/problems/implement-stack-using-queues/
用两个队列实现栈,首先明确栈和队列的特点,栈–后进先出,队列–先进先出。那么设置两个队列,设元素个数为n。
- 入数据,入数据到一个不为空队列,如果两个队列都为空,入哪一个都行。
- 出数据,将不为空的队列的前n-1个数据导入空的队列,输出最后一个数。
- 出入数据按照1和2执行,如此循环即可。
使用C语言的话,前面需要导入队列的实现
typedef struct {
//两个队列
Queue q1;
Queue q2;
} MyStack;
MyStack* myStackCreate() {
//函数返回值为栈的指针,那么需要动态开辟空间
MyStack* st=(MyStack*)malloc(sizeof(MyStack));
//分别初始化
QueueInit(&(st->q1));
QueueInit(&(st->q2));
return st;
}
void myStackPush(MyStack* obj, int x) {
//栈入数据,入数据到一个不为空队列,如果两个队列都为空,入哪一个都行。
if(!QueueEmpty(&(obj->q1)))
{
QueuePush(&(obj->q1),x);
}
else
{
QueuePush(&(obj->q2),x);
}
}
int myStackPop(MyStack* obj) {
//出数据,将不为空的队列的前n-1个数据导入空的队列,输出最后一个数
Queue* emptyq=&(obj->q1);
Queue* nonemptyq=&(obj->q2);
if(!QueueEmpty(&(obj->q1)))
{
emptyq=&(obj->q2);
nonemptyq=&(obj->q1);
}
//找到了哪个不是空的,哪个是空的
while(QueueSize(nonemptyq)>1)//留一个元素
{
//开倒
QueuePush(emptyq,QueueFront(nonemptyq));
QueuePop(nonemptyq);
}
//最后一个即是需要的输出的
int top=QueueFront(nonemptyq);
QueuePop(nonemptyq);
return top;
}
int myStackTop(MyStack* obj) {
//获取栈顶元素,其实就是非空队列的组后一个元素
Queue* emptyq=&(obj->q1);
Queue* nonemptyq=&(obj->q2);
if(!QueueEmpty(&(obj->q1)))
{
emptyq=&(obj->q2);
nonemptyq=&(obj->q1);
}
return QueueBack(nonemptyq);
}
bool myStackEmpty(MyStack* obj) {
//两个队列都为空才是空
return QueueEmpty(&(obj->q1))&&QueueEmpty(&(obj->q2));
}
void myStackFree(MyStack* obj) {
//先释放队列,再释放栈
QueueDestroy(&(obj->q1));
QueueDestroy(&(obj->q2));
free(obj);
}
https://leetcode-cn.com/problems/implement-queue-using-stacks/
同样是用两个栈实现一个队列,需牢记它们的各自的特点,与上题有略微不同,我们设置一个入栈pushST和一个出栈popST
- 入数据,只往pushST入
- 出数据,如果popST不为空,直接出栈;如果popST为空,将pushST数据倒入,再出栈
画图分析
使用C语言实现的话,前面也需实现栈
typedef struct {
//一个入数据栈,一个出数据栈
ST pushST;
ST popST;
} MyQueue;
int myQueuePeek(MyQueue* obj);
MyQueue* myQueueCreate() {
MyQueue* q=(MyQueue*)malloc(sizeof(MyQueue));
//分别初始化
StackInit(&(q->pushST));
StackInit(&(q->popST));
return q;
}
void myQueuePush(MyQueue* obj, int x) {
//入数据,只往pushST入
StackPush(&(obj->pushST),x);
}
int myQueuePop(MyQueue* obj) {
// //出数据,如果popST不为空,直接出栈;如果popST为空,将pushST数据倒入,再出栈
// if(StackEmpty(&(obj->popST)))
// {
// //为空
// //开倒
// while(!StackEmpty(&(obj->pushST)))
// {
// StackPush(&(obj->popST),StackTop(&(obj->pushST)));
// StackPop(&(obj->pushST));
// }
// }
//可直接调用peek函数倒数据
int front=myQueuePeek(obj);
//不为空就出栈
StackPop(&(obj->popST));
return front;
}
int myQueuePeek(MyQueue* obj) {
//返回队头元素
//如果popST为空,将pushST数据倒入,再出栈
if(StackEmpty(&(obj->popST)))
{
//为空
//开倒
while(!StackEmpty(&(obj->pushST)))
{
StackPush(&(obj->popST),StackTop(&(obj->pushST)));
StackPop(&(obj->pushST));
}
}
return StackTop(&(obj->popST));
}
bool myQueueEmpty(MyQueue* obj) {
return StackEmpty(&(obj->pushST))&&StackEmpty(&(obj->popST));
}
void myQueueFree(MyQueue* obj) {
StackDestroy(&(obj->pushST));
StackDestroy(&(obj->popST));
free(obj);
}
https://leetcode-cn.com/problems/design-circular-queue/
- 用顺序表实现
- 用循环链表实现
需要注意的是如何判断满和空
typedef int CQDataType;
typedef struct
{
int* a;//数组
int front;//头
int rear;//尾
int capacity;//容量
} MyCircularQueue;
bool myCircularQueueIsEmpty(MyCircularQueue* obj);
bool myCircularQueueIsFull(MyCircularQueue* obj);
//创建循环队列
MyCircularQueue* myCircularQueueCreate(int k)
{
MyCircularQueue* cq = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));
if (cq == NULL)
exit(-1);
cq->a = (CQDataType*)malloc(sizeof(CQDataType) * (k + 1));
//多开辟一个空间,用以区分满和空
if (cq->a == NULL)
exit(-1);
cq->capacity = k ;
cq->front = cq->rear = 0;
return cq;
}
//入队列
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value)
{
//判断是否满
if (myCircularQueueIsFull(obj))
return false;
//插入
obj->a[obj->rear] = value;
obj->rear = (obj->rear + 1) % (obj->capacity + 1);
return true;
}
//出队列
bool myCircularQueueDeQueue(MyCircularQueue* obj)
{
if (myCircularQueueIsEmpty(obj))
{
//为空
return false;
}
obj->front = (obj->front + 1) % (obj->capacity + 1);
return true;
}
//取队头元素
int myCircularQueueFront(MyCircularQueue* obj)
{
if(myCircularQueueIsEmpty(obj))
return -1;
return obj->a[obj->front];
}
//取队尾元素
int myCircularQueueRear(MyCircularQueue* obj)
{
if(myCircularQueueIsEmpty(obj))
return -1;
//取rear的前一个
return obj->a[(obj->rear + obj->capacity) % (obj->capacity + 1)];
}
//判断是否为空
bool myCircularQueueIsEmpty(MyCircularQueue* obj)
{
if (obj->front == obj->rear)
return true;
else
return false;
}
//判断队列是否满
bool myCircularQueueIsFull(MyCircularQueue* obj)
{
if ((obj->rear + 1) % (obj->capacity + 1) == obj->front)
return true;
else
return false;
}
//销毁队列
void myCircularQueueFree(MyCircularQueue* obj)
{
free(obj->a);
obj->capacity = obj->front = obj->rear = 0;
free(obj);
obj = NULL;
}
//链表实现
typedef int CQDataType;
typedef struct MyCircularQueueNode
{
CQDataType data;
struct MyCircularQueueNode* next;
}MyCQNode;
typedef struct
{
MyCQNode* head;//指向循环队列的头的前一个
MyCQNode* tail;//指向尾
int capacity;//容量
} MyCircularQueue;
//创建循环队列
MyCircularQueue* myCircularQueueCreate(int k);
//入队列
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value);
//出队列
bool myCircularQueueDeQueue(MyCircularQueue* obj);
//取队头元素
int myCircularQueueFront(MyCircularQueue* obj);
//取队尾元素
int myCircularQueueRear(MyCircularQueue* obj);
//判断是否为空
bool myCircularQueueIsEmpty(MyCircularQueue* obj);
//判断队列是否满
bool myCircularQueueIsFull(MyCircularQueue* obj);
//销毁队列
void myCircularQueueFree(MyCircularQueue* obj);
//链表实现
//创建循环队列
MyCircularQueue* myCircularQueueCreate(int k)
{
MyCircularQueue* cq = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));
if (cq == NULL)
exit(-1);
cq->capacity = k;
//开辟k+1个结点
//先一个
MyCQNode* head = (MyCQNode*)malloc(sizeof(MyCQNode));
if (head == NULL)
exit(-1);
head->next = NULL;
MyCQNode* sur = head;
while (k--)
{
//再开K个
MyCQNode* newnode = (MyCQNode*)malloc(sizeof(MyCQNode));
if (newnode == NULL)
exit(-1);
newnode->next = NULL;
sur->next = newnode;
sur = sur->next;
}
sur->next = head;//循环链表
cq->head = cq->tail = head;//最初都指向第一个
return cq;
}
//入队列
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value)
{
if (myCircularQueueIsFull(obj))
return false;
//tail指向的下一个位置加入
obj->tail->next->data = value;
obj->tail = obj->tail->next;
return true;
}
//出队列
bool myCircularQueueDeQueue(MyCircularQueue* obj)
{
if (myCircularQueueIsEmpty(obj))
return false;
obj->head = obj->head->next;
return true;
}
//取队头元素
int myCircularQueueFront(MyCircularQueue* obj)
{
if (myCircularQueueIsEmpty(obj))
return -1;
//head指向的下一个结点数据
return obj->head->next->data;
}
//取队尾元素
int myCircularQueueRear(MyCircularQueue* obj)
{
if (myCircularQueueIsEmpty(obj))
return -1;
return obj->tail->data;
}
//判断是否为空
bool myCircularQueueIsEmpty(MyCircularQueue* obj)
{
if (obj->head == obj->tail)
return true;
else
return false;
}
//判断队列是否满
bool myCircularQueueIsFull(MyCircularQueue* obj)
{
if (obj->head == obj->tail->next)
return true;
else
return false;
}
//销毁队列
void myCircularQueueFree(MyCircularQueue* obj)
{
MyCQNode* temp = obj->head;
MyCQNode* headnext = obj->head->next;
temp->next = NULL;//断链
while (headnext != NULL)
{
temp = headnext->next;
free(headnext);
headnext = temp;
}
obj->capacity = 0;
obj->head = obj->tail = NULL;
free(obj);
obj = NULL;
}