class="hide-preCode-box">
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
5.返回队头元素
返回队头元素的逻辑跟上面的出队列操作差不多,如果stpop不为空那么直接返回栈顶元素(队头元素),如果stpop为空那么就先把stpush中的元素移动到stpop栈中,然后再去返回栈顶元素 跟上面出栈操作的唯一区别就是不用出栈,只用去取栈顶元素 ,如下:
int myQueuePeek ( MyQueue* obj)
{
if ( STEmpty ( & obj-> stpop) )
{
while ( ! STEmpty ( & obj-> stpush) )
{
int top = STTop ( & obj-> stpush) ;
STPop ( & obj-> stpush) ;
STPush ( & obj-> stpop, top) ;
}
}
return STTop ( & obj-> stpop) ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
6.队列判空和队列销毁
我们用两个栈实现的队列要判空的话,必须保证两个栈同时为空队列才为空,否则就不是空,如下:
bool myQueueEmpty ( MyQueue* obj)
{
return STEmpty ( & obj-> stpush) && STEmpty ( & obj-> stpop) ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
队列销毁的本质也就是销毁里面的两个栈,但是由于我们的MyQueue这个指针也是我们动态申请的,也要释放掉,然后置空,如下:
void myQueueFree ( MyQueue* obj)
{
STDestroy ( & obj-> stpush) ;
STDestroy ( & obj-> stpop) ;
free ( obj) ;
obj = NULL ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
7.完整题解代码
那么关于栈的代码,大家就可以自行拷贝了,在这里拷贝也显得冗余了
typedef struct
{
ST stpush;
ST stpop;
} MyQueue;
MyQueue* myQueueCreate ( )
{
myQueue* pq = ( myQueue* ) malloc ( sizeof ( myQueue) ) ;
if ( pq == NULL )
{
perror ( "malloc fail\n" ) ;
return NULL ;
}
STInit ( & pq-> stpush) ;
STInit ( & pq-> stpop) ;
}
void myQueuePush ( MyQueue* obj, int x)
{
STPush ( & obj-> stpush, x) ;
}
int myQueuePop ( MyQueue* obj)
{
if ( STEmpty ( & obj-> stpop) )
{
while ( ! STEmpty ( & obj-> stpush) )
{
int top = STTop ( & obj-> stpush) ;
STPop ( & obj-> stpush) ;
STPush ( & obj-> stpop, top) ;
}
}
int top = S TTop ( & obj-> stpop) ;
STPop ( & obj-> stpop) ;
return top;
}
int myQueuePeek ( MyQueue* obj)
{
if ( STEmpty ( & obj-> stpop) )
{
while ( ! STEmpty ( & obj-> stpush) )
{
int top = STTop ( & obj-> stpush) ;
STPop ( & obj-> stpush) ;
STPush ( & obj-> stpop, top) ;
}
}
return STTop ( & obj-> stpop) ;
}
bool myQueueEmpty ( MyQueue* obj)
{
return STEmpty ( & obj-> stpush) && STEmpty ( & obj-> stpop) ;
}
void myQueueFree ( MyQueue* obj)
{
STDestroy ( & obj-> stpush) ;
STDestroy ( & obj-> stpop) ;
free ( obj) ;
obj = NULL ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68
二、用队列实现栈
题目链接:用队列实现栈 还是先来看看题目:
1.大致思路分析
这道题和上道题类似,就是使用两个队列实现一个栈的各种操作,这道题要比上一题难,因为在上一题中,我们的思路就是“负负得正”,通过两个栈的互相倒腾就可以还原队列的入队列出队列操作,但是这题不行,我们画个图看看:
根据上图的分析,我们知道了完全使用上一题的思路行不通,因为栈通过两次倒腾后,里面的元素顺序会反过来,但是队列通过两次倒腾后,里面元素的顺序还是不变
所以我们要换个思路,我们先来看看出栈的思路: 我们把数据入到队列1中,然后移动到队列2时,不全部挪过去,只留下一个,如图:
这个时候我们发现队列1中留下的,对应到栈上不正是我们的栈顶元素吗?我们在出栈的时候就可以把队列1中剩下的一个元素出队列,4是最后入栈的,最后第一个出栈,符合我们栈的性质
那么为了保险起见,我们再试一次,先假设我们实现的栈出栈一次,就是把4删掉,让剩下的1,2,3再来模拟一次出栈,如图:
此时我们发现队列2中留下的正好是新的栈顶元素,让队列2出队列就刚好对应了出栈操作,所以我们的出栈思路大致是正确的
但是我们发现,队列1和队列2是反复倒腾的,没有哪个是专门出栈,哪个专门入栈的,所以我们接下来要分清怎么判断哪个队列什么时候用来入栈,什么时候用来出栈
经过规律总结我们发现,当我们要出栈的时候,有一个队列总是空着的,另一个队列里面才装有数据,我们在操作时就是让不为空的队列将size-1个元素挪到空队列里面去,然后不为空的队列就只剩下一个元素出队列即可模拟实现出栈
那么我们入栈的时候,就是往不为空的队列中去入,所以我们在写入栈和出栈操作时最重要的就时判断出哪个队列空,哪个队列不为空,才方便进行操作,那么接下来我们就一个函数一个函数分析
2.用队列实现栈的结构和初始化
既然题目明确要求我们用两个队列实现一个栈,所以栈的结构实际上就是两个队列,如下:
typedef struct
{
Queue q1;
Queue q2;
} MyStack;
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
初始化也很简单,首先动态开辟一个MyStack大小的空间,然后就只需要调用队列的初始化方法分别初始化两个队列即可,如下:
MyStack* myStackCreate ( )
{
MyStack* st = ( MyStack* ) malloc ( sizeof ( MyStack) ) ;
if ( st == NULL )
{
perror ( "malloc fail\n" ) ;
return NULL ;
}
QueueInit ( & st-> q1) ;
QueueInit ( & st-> q2) ;
return st;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
3.入栈
我们刚刚分析过了,我们入栈就是往不为空的队列中去入,所以我们在入栈时只需要判断一下哪个栈不为空即可,如下:
void myStackPush ( MyStack* obj, int x)
{
if ( ! QueueEmpty ( & obj-> q1) )
{
QueuePush ( & obj-> q1, x) ;
}
else
{
QueuePush ( & obj-> q2, x) ;
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
4.出栈
我们的出栈操作刚刚也重点分析了,就是先找出哪个是空队列,哪个是非空队列,然后将非空队列中size-1个数据挪动到空队列中,然后非空队列剩下的一个元素就是我们要出栈的元素,题目要求出栈前返回栈顶元素,所以我们创建一个变量用来保存栈顶元素,最后用来返回即可,如下:
int myStackPop ( MyStack* obj)
{
Queue* empty = & obj-> q1;
Queue* noempty = & obj-> q2;
if ( ! QueueEmpty ( & obj-> q1) )
{
empty = & obj-> q2;
noempty = & obj-> q1;
}
int sum = QueueSize ( noempty) - 1 ;
while ( sum-- )
{
int top = QueueFront ( noempty) ;
QueuePop ( noempty) ;
QueuePush ( empty, top) ;
}
int top = QueueFront ( noempty) ;
QueuePop ( noempty) ;
return top;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
5.取栈顶元素
按照我们上面的思想,取栈顶元素其实就是取非空队列中的队尾元素,如下:
int myStackTop ( MyStack* obj)
{
if ( ! QueueEmpty ( & obj-> q1) )
{
return QueueBack ( & obj-> q1) ;
}
else
{
return QueueBack ( & obj-> q2) ;
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
6.判空和销毁栈
只有两个队列同时为空,我们的栈才为空,如下:
bool myStackEmpty ( MyStack* obj)
{
return QueueEmpty ( & obj-> q1) && QueueEmpty ( & obj-> q2) ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
销毁栈就只需要调用队列的销毁函数,分别销毁掉两个队列即可,不要忘了最后把整个栈的结构也销毁掉,如下:
void myStackFree ( MyStack* obj)
{
QueueDestroy ( & obj-> q1) ;
QueueDestroy ( & obj-> q2) ;
free ( obj) ;
obj = NULL ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
7.完整题解
typedef int QDataType;
typedef struct QueueNode
{
QDataType data;
struct QueueNode * next;
} QueueNode;
typedef struct Queue
{
int size;
QueueNode* phead;
QueueNode* ptail;
} Queue;
void QueueInit ( Queue* pq)
{
assert ( pq) ;
pq-> size = 0 ;
pq-> phead = pq-> ptail = NULL ;
}
void QueueDestroy ( Queue* pq)
{
assert ( pq) ;
QueueNode* pcur = pq-> phead;
while ( pcur)
{
QueueNode* del = pcur;
pcur = pcur-> next;
free ( del) ;
del = NULL ;
}
pq-> phead = pq-> ptail = NULL ;
pq-> size = 0 ;
}
QueueNode* QueueBuyNode ( QDataType x)
{
QueueNode* newnode = ( QueueNode* ) malloc ( sizeof ( QueueNode) ) ;
if ( newnode == NULL )
{
perror ( "malloc fail\n" ) ;
return NULL ;
}
newnode-> data = x;
newnode-> next = NULL ;
return newnode;
}
void QueuePush ( Queue* pq, QDataType x)
{
assert ( pq) ;
QueueNode* newnode = QueueBuyNode ( x) ;
if ( pq-> phead == NULL )
{
pq-> phead = pq-> ptail = newnode;
pq-> size++ ;
return ;
}
pq-> ptail-> next = newnode;
pq-> ptail = newnode;
pq-> size++ ;
}
bool QueueEmpty ( Queue* pq)
{
assert ( pq) ;
return pq-> phead == NULL ;
}
void QueuePop ( Queue* pq)
{
assert ( ! QueueEmpty ( pq) ) ;
QueueNode* next = pq-> phead-> next;
free ( pq-> phead) ;
pq-> phead = next;
pq-> size-- ;
}
QDataType QueueFront ( Queue* pq)
{
assert ( pq) ;
return pq-> phead-> data;
}
QDataType QueueBack ( Queue* pq)
{
assert ( pq) ;
return pq-> ptail-> data;
}
int QueueSize ( Queue* pq)
{
assert ( pq) ;
return pq-> size;
}
typedef struct
{
Queue q1;
Queue q2;
} MyStack;
MyStack* myStackCreate ( )
{
MyStack* st = ( MyStack* ) malloc ( sizeof ( MyStack) ) ;
if ( st == NULL )
{
perror ( "malloc fail\n" ) ;
return NULL ;
}
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)
{
Queue* empty = & obj-> q1;
Queue* noempty = & obj-> q2;
if ( ! QueueEmpty ( & obj-> q1) )
{
empty = & obj-> q2;
noempty = & obj-> q1;
}
int sum = QueueSize ( noempty) - 1 ;
while ( sum-- )
{
int top = QueueFront ( noempty) ;
QueuePop ( noempty) ;
QueuePush ( empty, top) ;
}
int top = QueueFront ( noempty) ;
QueuePop ( noempty) ;
return top;
}
int myStackTop ( MyStack* obj)
{
if ( ! QueueEmpty ( & obj-> q1) )
{
return QueueBack ( & obj-> q1) ;
}
else
{
return QueueBack ( & obj-> q2) ;
}
}
bool myStackEmpty ( MyStack* obj)
{
return QueueEmpty ( & obj-> q1) && QueueEmpty ( & obj-> q2) ;
}
void myStackFree ( MyStack* obj)
{
QueueDestroy ( & obj-> q1) ;
QueueDestroy ( & obj-> q2) ;
free ( obj) ;
obj = NULL ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
class="hide-preCode-box">1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
OK,那么今天的刷题也就到这里了,咱们下一篇再见啦~~
data-report-view="{"mod":"1585297308_001","spm":"1001.2101.3001.6548","dest":"https://blog.csdn.net/2401_88325505/article/details/145270943","extend1":"pc","ab":"new"}">>
评论记录:
回复评论: