第一章:栈
1.1栈的概念及结构
栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。
压栈:栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶。
出栈:栈的删除操作叫做出栈。出数据也在栈顶。
1.2 栈的实现
栈的实现一般可以使用数组或者链表实现,相对而言数组的结构实现更优一些。因为数组在尾上插入数据的代价比较小。
栈头文件各函数声明
#pragma once
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
typedef int STDataType;
typedef struct Stack//数组栈
{
STDataType* a;
int top;//栈顶
int capacity;//容量
}ST;
void STInit(ST* pst);//初始化栈
void STDestroy(ST* pst);//销毁栈
void STPush(ST* pst, STDataType x);//入栈
void STPop(ST* pst);//出栈
STDataType STTop(ST* pst);//获取栈顶元素
bool STEmpty(ST* pst);//栈是否为空
int STSize(ST* pst);//栈中有效元素个数
栈各函数实现
初始化栈
void STInit(ST* pst)//初始化栈
{
assert(pst);
pst->a = NULL;
pst->top = 0; //该初始化方式,栈顶指向【栈顶元素的后面】。先插入,再++
//pst->top = -1;//该初始化方式,栈顶指向【栈顶元素】。先++,再插入
pst->capacity = 0;
}
栈是否为空
bool STEmpty(ST* pst)//栈是否为空
{
assert(pst);
return pst->top == 0;//为空返回true;不为空返回false
}
入栈
void STPush(ST* pst, STDataType x)//入栈
{
assert(pst);
if (pst->top == pst->capacity)//检查容量(如果栈顶等于容量)
{
int newCapacity = pst->capacity == 0 ? 4 : 2 * pst->capacity;//栈容量为0就开辟4个单位,否则开辟原容量2倍
STDataType* tmp = (STDataType*)realloc(pst->a, sizeof(STDataType) * newCapacity);
if (tmp == NULL)
{
perror("realloc fail");
return;
}
pst->a = tmp;
pst->capacity = newCapacity;
}
pst->a[pst->top] = x;
pst->top++;
}
出栈
void STPop(ST* pst)//出栈
{
assert(pst);
assert(!STEmpty(pst));
pst->top--;
}
获取栈顶元素
STDataType STTop(ST* pst)//获取栈顶元素
{
assert(pst);
assert(!STEmpty(pst));
return pst->a[pst->top - 1];
}
栈中有效元素个数
int STSize(ST* pst)//栈中有效元素个数
{
assert(pst);
return pst->top;
}
销毁栈
void STDestroy(ST* pst)//销毁栈
{
assert(pst);
free(pst->a);
pst->a = NULL;
pst->top = pst->capacity = 0;
}
栈的测试
void TestStack1()
{
ST st;
STInit(&st);
STPush(&st, 1);
STPush(&st, 2);
printf("%d ", STTop(&st));
STPop(&st);
STPush(&st, 3);
STPush(&st, 4);
while (!STEmpty(&st))
{
printf("%d ", STTop(&st));
STPop(&st);
}
STDestroy(&st);
}
int main()
{
TestStack1();//2 4 3 1
return 0;
}
第二章:队列
2.1 队列的概念及结构
队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出FIFO(First In First Out) 。
入队列:进行插入操作的一端称为队尾。
出队列:进行删除操作的一端称为队头。
2.2 队列的实现
队列也可以数组和链表的结构实现,使用链表的结构实现更优一些,因为如果使用数组的结构,出队列在数组头上出数据,效率会比较低。
队里头文件各函数声明
#pragma once
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
typedef int QDataType;
typedef struct QueueNode //队列节点
{
struct QueueNode* next;
QDataType data;
}QNode;
typedef struct Queue //储存队列节点头尾指针及节点个数的结构体
{
QNode* phead;
QNode* ptail;
int size;
}Queue;
void QueueInit(Queue* pq);//队列初始化
void QueueDestroy(Queue* pq);//队列释放
void QueuePush(Queue* pq, QDataType x);//队列插入
void QueuePop(Queue* pq);//队列删除
QDataType QueueFront(Queue* pq);//取队头数据
QDataType QueueBack(Queue* pq);//取队尾数据
int QueueSize(Queue* pq);//队列元素个数
bool QueueEmpty(Queue* pq);//队列是否为空
队列各函数实现
初始化队列
void QueueInit(Queue* pq)//初始化队列
{
assert(pq);
pq->phead = NULL;
pq->ptail = NULL;
pq->size = 0;
}
队列是否为空
队列的插入、删除、获取队头队尾数据需要判断是否为空
//队列的插入、删除、获取队头队尾数据需要判断是否为空
bool QueueEmpty(Queue* pq)//队列是否为空
{
assert(pq);
return pq->phead == NULL && pq->ptail == NULL;
}
队尾入队列
void QueuePush(Queue* pq, QDataType x)//队列插入
{
assert(pq);
//创建队列节点
QNode* newnode = (QNode*)malloc(sizeof(QNode));
if (newnode == NULL)
{
perror("malloc newnode fail");
return;
}
newnode->data = x;
newnode->next = NULL;
//插入
if (QueueEmpty(pq) == true)
pq->phead = pq->ptail = newnode;
else //非空队列
{
pq->ptail->next = newnode;
pq->ptail = newnode;
}
pq->size++;
}
队头出队列
void QueuePop(Queue* pq)//队列删除
{
assert(pq);
assert(!QueueEmpty(pq));
if (pq->phead->next == NULL) //1.一个节点
{
free(pq->phead);
pq->phead = pq->ptail = NULL;
}
else //2.多个节点
{ //头删
QNode* next = pq->phead->next;
free(pq->phead);
pq->phead = next;
}
pq->size--;
}
获取队列头部元素
QDataType QueueFront(Queue* pq)//取队头数据
{
assert(pq);
assert(!QueueEmpty(pq));
return pq->phead->data;
}
获取队列队尾元素
QDataType QueueBack(Queue* pq)//取队尾数据
{
assert(pq);
assert(!QueueEmpty(pq));
return pq->ptail->data;
}
获取队列中有效元素个数
int QueueSize(Queue* pq)//队列元素个数
{
assert(pq);
return pq->size;
}
释放队列
void QueueDestroy(Queue* pq)//释放队列
{
assert(pq);
QNode* cur = pq->phead;
while (cur)
{
QNode* del = cur;
cur = cur->next;
free(del);
}
pq->phead = pq->ptail = NULL;
pq->size = 0;
}
队列测试
#include "Queue.h"
#include <stdio.h>
void TestQueue()
{
Queue q;
QueueInit(&q);
QueuePush(&q, 1);
QueuePush(&q, 2);
printf("%d ", QueueFront(&q));
QueuePop(&q);
QueuePush(&q, 3);
QueuePush(&q, 4);
printf("Size:%d\n", QueueSize(&q));
while (!QueueEmpty(&q))
{
printf("%d ", QueueFront(&q));
QueuePop(&q);
}
QueueDestroy(&q);
}
int main()
{
TestQueue();
return 0;
}
第三章:栈和队列面试题
1. 括号匹配问题
给定一个只包括 '(' ,')' ,'{' ,'}' ,'[' ,']' 的字符串 s ,判断字符串是否有效。
有效字符串需满足:
- 左括号必须用相同类型的右括号闭合。
- 左括号必须以正确的顺序闭合。
- 每个右括号都有一个对应的相同类型的左括号。
注意:此题需要用到上面栈实现的各函数
//1. 创建并初始化栈
//2. 数组指针遇到左括号入栈
//3. 数组指针遇到右括号份2种情况:
// a. 栈为空,返回假。(说明之前没有遇到左括号)
// b. 栈不为空,获取栈顶元素与数组指针指向元素比较是否匹配。(注意匹配条件,找为假的情况)
//重复上述步骤直到遍历完数组。
//4.最后判断栈是否为空。空为真,非空为假。
bool isValid(char* s) {
ST st;// 创建栈
STInit(&st); // 初始化栈
// 1.左括号入栈
// 2.出栈元素与右括号匹配
while (*s) { // 遍历数组
if (*s == '(' || *s == '[' || *s == '{') // 左括号入栈
STPush(&st, *s);
else { // 出栈元素与右括号匹配
// 在出栈之前要判断栈是否为空,如果为空,说明没有左括号,直接返回假
if (STEmpty(&st)) {
STDestroy(&st);
return false;
}
char top = STTop(&st); // 获取栈顶元素
STPop(&st); // 将栈顶元素出栈
//if ((*s == ')' && top == '(') || (*s == ']' && top == '[') || (*s == '}' && top == '{'))
//这里不能用上方条件判断正确后直接返回true,因为最后一个元素为左括号时也满足,但不正确
//下方如果栈顶元素与数组指针指向元素不匹配返回假
if ((*s == ')' && top != '(') || (*s == ']' && top != '[') || (*s == '}' && top != '{')) {
STDestroy(&st);
return false;
}
}
s++;
}
//这里判断栈里是否还有元素,
// 如果有,说明有一个单独的左括号,不匹配
// 如果没有,说明都匹配完成
bool ret = STEmpty(&st);
STDestroy(&st);
return ret;
}
2. 用队列实现栈
请你仅使用两个队列实现一个后入先出(LIFO)的栈,并支持普通栈的全部四种操作(push
、top
、pop
和 empty
)。(此题需要用到上方队列实现的各函数)
// 往不为空的队列插入数据,当需要弹出时,将n-1个数据插入到另外一个队列,此时需要弹出的数据就在该队列最前端。反之亦然
typedef struct {
// Queue为储存队列节点头尾指针及节点个数的结构体
Queue q1, q2;//创建2个用于维护两个队列
// Queue *q1, *q2;
//这样创建不好,因为未初始化,所以是野指针,当下方调用QueueInit函数初始化时,就会造成对野指针的解引用
//除非使用malloc开辟Queue结构体的空间。而且后续使用完还需要释放。
// 不如创建2个Queue结构体封装到MyStack结构体中,直接malloc开辟MyStack需要的空间
} MyStack;
// 两个队列在操作完数据后,一定是某一个为空,或者两个都为空
MyStack* myStackCreate() {
MyStack* obj = (MyStack*)malloc(sizeof(MyStack));
if (obj == NULL) {
perror("malloc MyStack* obj fail");
return NULL;
}
QueueInit(&obj->q1);
QueueInit(&obj->q2);
return obj;
}
void myStackPush(MyStack* obj, int x) {
if (!QueueEmpty(&obj->q1)) //当q1不为空时,往q1入数据
QueuePush(&obj->q1, x);
else
QueuePush(&obj->q2, x); //只要q1有数据,就可以向q2插入数据
}
int myStackPop(MyStack* obj) { //栈的pop是后进先出
// 弹出数据时就是在两个队列相互导数据,所以要判断哪个队列为空
// 1.创建名为空和非空的变量。假设q1为空,q2不为空,如果假设错误,互换。
// 2.将非空队列的数据导入空队列。非空队列数据剩余1时,该数据就是要弹出的数据
//第一步
Queue* pEmptyQ = &obj->q1;
Queue* pNonEmptyQ = &obj->q2;
if (!QueueEmpty(&obj->q1)) {
pEmptyQ = &obj->q2;
pNonEmptyQ = &obj->q1;
}
// 第二步
while (QueueSize(pNonEmptyQ) > 1) { //非空队列数据还剩1个时就是要弹出的数据
QueuePush(pEmptyQ, QueueFront(pNonEmptyQ));//非空队列取队头数据插入到空队列
QueuePop(pNonEmptyQ);//弹出非空队列数据
}
int top = QueueFront(pNonEmptyQ);//此时非空队列还剩1个数据,取非空队列队头数据
QueuePop(pNonEmptyQ);//再弹出非空队列的数据
return top;
}
int myStackTop(MyStack* obj) {
if (!QueueEmpty(&obj->q1)) //如果q1队列不为空,那么返回q1队列的队尾数据
return QueueBack(&obj->q1);
else
return QueueBack(&obj->q2);//如果q2队列不为空,那么返回q2队列的队尾数据
}
bool myStackEmpty(MyStack* obj) {
return QueueEmpty(&obj->q1) && QueueEmpty(&obj->q2);
}
void myStackFree(MyStack* obj) {
QueueDestroy(&obj->q1);//q1或q2队列是否为空都需要释放。为空只是没有数据,但队列已经申请了空间
QueueDestroy(&obj->q2);
free(obj);
//不能直接释放obj,因为obj里面是两个维护【队列头尾指针和节点个数】的结构体。
// 释放obj只是释放了这两个结构体的空间,但是队列的节点还未释放
}
3. 用栈实现队列
请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作(push
、pop
、peek
、empty
)。(此题需要用到上方栈实现的各函数)
typedef struct {
ST pushst;
ST popst;
} MyQueue;
MyQueue* myQueueCreate() {
MyQueue* obj = (MyQueue*)malloc(sizeof(MyQueue));
if (obj == NULL) {
perror("malloc MyQueue* obj fail");
return NULL;
}
STInit(&obj->pushst);
STInit(&obj->popst);
return obj;
}
void myQueuePush(MyQueue* obj, int x) {
STPush(&obj->pushst, x);
}
int myQueuePeek(MyQueue* obj) {
if (STEmpty(&obj->popst)) {
while (!STEmpty(&obj->pushst)) {
STPush(&obj->popst, STTop(&obj->pushst));
STPop(&obj->pushst);
}
}
return STTop(&obj->popst);
}
int myQueuePop(MyQueue* obj) {
int front = myQueuePeek(obj);
STPop(&obj->popst);
return front;
}
bool myQueueEmpty(MyQueue* obj) {
return STEmpty(&obj->pushst) && STEmpty(&obj->popst);
}
void myQueueFree(MyQueue* obj) {
STDestroy(&obj->pushst);
STDestroy(&obj->popst);
free(obj);
}
4. 设计循环队列
思路:
用数组队列,front指向头元素,rear指向尾元素的后面。
假设有k个数据,那么开辟k+1个空间,因为rear需要指向尾元素后面
注意:rear不能指向尾元素,否则只有一个元素时,或满元素时,front和rear都指向同一位置,无法判断满还是空
typedef struct {
int front; // 队列头下标
int rear; // 队列尾下标
int k; // 元素个数(但开辟空间要比数据多一个)
int* a; // 数组
} MyCircularQueue;
bool myCircularQueueIsEmpty(MyCircularQueue* obj) { //队列是否为空
return obj->front == obj->rear;//头尾下标相等即为空
}
bool myCircularQueueIsFull(MyCircularQueue* obj) { //队列是否满
return (obj->rear + 1) % (obj->k + 1) == obj->front;
// 因为rear指向最后元素后面,又因队列空间比满元素多1。
// 即rear指向的是队列最后一个空间,所以rear+1就和front指向同一位置。
// 因为是循环队列,所以rear+1要%队列空间个数
}
MyCircularQueue* myCircularQueueCreate(int k) { //创建队列
MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));//开辟维护队列信息结构体的空间
if (obj == NULL) {
perror("malloc MyCircularQueue* obj fail");
return NULL;
}
obj->a = (int*)malloc(sizeof(int) * (k + 1));//开辟队列
obj->front = obj->rear = 0;//头尾下标指向同一位置(也就是空队列)
obj->k = k;//队列元素个数
return obj;
}
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) { //插入数据
if (myCircularQueueIsFull(obj))//队列满返回false
return false;
obj->a[obj->rear++] = value;//在rear位置处插入数据,rear++
//因为循环队列,rear可能回到数组开头,为确保索引在 0 到 obj->k 之间循环,所以需要取模处理。
obj->rear %= (obj->k + 1);
return true;
}
bool myCircularQueueDeQueue(MyCircularQueue* obj) { //删除数据
if (myCircularQueueIsEmpty(obj))//队列为空返回false
return false;
obj->front++;//移动头下标即可
obj->front %= (obj->k + 1);//因为循环队列,头下标同上方尾下标同理,需要取模处理
return true;
}
int myCircularQueueFront(MyCircularQueue* obj) { //获取队首元素
if (myCircularQueueIsEmpty(obj))//队列为空返回-1
return -1;
return obj->a[obj->front];//队列不为空,直接返回队首元素
}
int myCircularQueueRear(MyCircularQueue* obj) { //获取队尾元素
if (myCircularQueueIsEmpty(obj))//队列为空返回-1
return -1;
// if (obj->rear == 0)
// return obj->a[obj->k];
// else
// return obj->a[obj->rear - 1];
//因为是循环队列,尾下标分2种情况
//1.尾下标在队列头。 2.尾下标不在队列头
return obj->a[(obj->rear - 1 + obj->k + 1) % (obj->k + 1)];
//obj->rear-1是队尾元素,obj->k+1是队列空间个数。
//队尾元素下标+队列空间个数相当于绕了一圈停在原来的位置。
//之所以要加上obj->k+1,是因为如果rear恰好指向队列头,那么-1操作就会rear就会变成负数。
//这一步相当于将索引值转化为非负数,以确保后续的取模操作不会得到负数结果
//所以该操作是为了确保在模运算中得到正确的结果
}
void myCircularQueueFree(MyCircularQueue* obj) { //释放队列
free(obj->a);//释放队列数组
free(obj);//释放维护队列信息的结构体
}
第四章:概念选择题
1.一个栈的初始状态为空。现将元素1、2、3、4、5、A、B、C、D、E依次入栈,然后再依次出栈,则元素出栈的顺序是( )。
A. 12345ABCDE
B. EDCBA54321
C. ABCDE12345
D. 54321EDCBA
答案:B
2.若进栈序列为 1,2,3,4 ,进栈过程中可以出栈,则下列不可能的一个出栈序列是()
A. 1, 4, 3, 2
B. 2, 3, 4, 1
C. 3, 1, 4, 2
D. 3, 4, 2, 1
答案:C
3.循环队列的存储空间为 Q(1:100) ,初始状态为 front=rear=100 。经过一系列正常的入队与退队操作后, front=rear=99 ,则循环队列中的元素个数为( )
A. 1
B. 2
C. 99
D. 0或者100
答案:D
4.以下( )不是队列的基本运算?
A. 从队尾插入一个新元素
B. 从队列中删除第i个元素
C. 判断一个队列是否为空
D. 读取队头元素的值
答案:B
5.现有一循环队列,其队头指针为front,队尾指针为rear;循环队列长度为N。其队内有效长度为?(假设队头不存放数据)
A. (rear - front + N) % N + 1
B. (rear - front + N) % N
C. (rear - front) % (N + 1)
D. (rear - front + N) % (N - 1)
答案:B
解析:有效长度一般是rear-front, 但是循环队列中rear有可能小于front,减完之后可能是负数,所以需要+N,此时结果刚好是队列中有效元素个数,但如果rear大于front,减完之后就是有效元素个数了,再加N后有效长度会超过N,故需要%N。
6. 下列关于顺序结构实现循环队列的说法,正确的是( )
A.循环队列的长度通常都不固定
B.直接用队头和队尾在同一个位置可以判断循环队列是否为满
C.通过设置计数的方式可以判断队列空或者满
D.循环队列是一种非线性数据结构
答案:C
队列适合使用链表实现,使用顺序结构(即固定的连续空间)实现时会出现假溢出的问题,因此大佬们设计出了循环队列,循环队列就是为了解决顺序结构实现队列假溢出问题的
A错误:循环队列的长度都是固定的
B错误:队头和队尾在同一个位置时 队列可能是空的,也可能是满的,因此无法判断
C正确:设置计数即添加一个字段来记录队列中有效元素的个数,如果队列中有效元素个数等于空间总大小时队列满,如果队列中有效元素个数为0时队列空
D错误:循环队列也是队列的一种,是一种特殊的线性数据结构
7. 下列关于栈的叙述正确的是( )
A.栈是一种“先进先出”的数据结构
B.栈可以使用链表或顺序表来实现
C.栈只能在栈底插入数据
D.栈不能删除数据
答案:B
A错误:栈是一种后进先出的数据结构,队列是先进先出的
B正确:顺序表和链表都可以用来实现栈,不过一般都使用顺序表,因为栈想当于是阉割版的顺序表,只用到了顺序表的尾插和尾删操作,顺序表的尾插和尾删不需要搬移元素效率非常高,故一般都是使用顺序表实现
C错误:栈只能在栈顶进行输入的插入和删除操作
D错误:栈是有入栈和出栈操作,出栈就是从栈中删除一个元素
8. 一个栈的入栈序列为ABCDE,则不可能的出栈序列为( )
A.ABCDE
B.EDCBA
C.DCEBA
D.ECDBA
答案:D
如果是E先出,说明ABCDE都已经全部入栈,E出栈之后,此时栈顶元素是D,如果再要出栈应该是D,而不应该是C。
9. 下列关于用栈实现队列的说法中错误的是( )
A.用栈模拟实现队列可以使用两个栈,一个栈模拟入队列,一个栈模拟出队列
B.每次出队列时,都需要将一个栈中的全部元素导入到另一个栈中,然后出栈即可
C.入队列时,将元素直接往模拟入队列的栈中存放即可
D.入队列操作时间复杂度为O(1)
答案:B
选项B中,一个栈模拟入队列,一个栈模拟出队列,出队列时直接弹出模拟出队列栈的栈顶元素,当该栈为空时,将模拟入队列栈中所有元素导入即可,不是每次都需要导入元素,故错误
选项A中,栈和队列的特性是相反的,一个栈实现不了队列
选项C中,一个栈模拟入队列,一个栈模拟出队列,入队列时,将元素直接往模拟入队列的栈中存放
选项D中,入队列就是将元素放到栈中,因此时间复杂度就是O(1)
10. 链栈与顺序栈相比,比较明显的优点是( )
A.插入操作更加方便
B.删除操作更加方便
C.入栈时不需要扩容
答案:C
A错误,如果是链栈,一般需要进行头插或者头删操作,而顺序栈一般进行尾插和尾删操作,链表的操作比顺序表复杂,因此使用顺序结构实现栈更简单
B错误,原因参考A
C正确,链式结构实现栈时,每次入栈相当于链表中头插一个节点,没有扩容一说
11. 下列关于队列的叙述错误的是( )
A.队列可以使用链表实现
B.队列是一种“先入先出”的数据结构
C.数据出队列时一定只影响尾指针
D.数据入队列时一定从尾部插入
答案:C
出队操作,一定会影响头指针,如果出队之后,队列为空,会影响尾指针。
12. 以下不是队列的基本运算的是( )
A.从队尾插入一个新元素
B.从队列中删除队尾元素
C.判断一个队列是否为空
D.读取队头元素的值
答案:B
队列只能从队头删除元素。
13. 下面关于栈和队列的说法中错误的是( )
A.队列和栈通常都使用链表实现
B.队列和栈都只能从两端插入、删除数据
C.队列和栈都不支持随机访问和随机插入
D.队列是“先入先出”,栈是“先入后出”
答案:A B
A错误:栈是尾部插入和删除,一般使用顺序表实现,队列是头部删除尾部插入,一般使用链表实现
B错误:栈是后进先出,尾部插入和删除;队列是先进先出,尾部插入头部删除
C正确:栈只能访问栈顶元素,不支持随机访问,队列也不支持
D正确:栈和队列的特性
14. 用无头单链表存储队列,其头指针指向队头结点,尾指针指向队尾结点,则在进行出队操作时( )
A.仅修改队头指针
B.队头、队尾指针都要修改
C.队头、队尾指针都可能要修改
D.仅修改队尾指针
答案:C
出队操作,一定会修改头指针,如果出队之后,队列为空,需要修改尾指针。