题目解读
本题是要求我们设计一个循环的队列,循环队列要有以下功能:
1.获取队首元素,若队列为空返回-1
2.获取队尾元素,若队列为空,则返回-1
3.插入元素,插入成功返回真
4.删除元素,删除成功返回真
5.检查队列是否为空
6.检查队列是否已满
首先我们可以将之前写的用链表实现的队列的代码拷贝到该题中,以便于循环队列的实现,然后开始构思。
循环队列的解释题目中也给出了解释:
循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环。它也被称为“环形缓冲器”。
解题构思
所以我们可以把循环队列先画图,他是一个环形的队列,并且首位相连尾接
那么,循环队列什么时候是满的,什么时候是空的呢?
其实,当队首和队尾在同一个位置时,这个时候队列就是空的,而当对头front的位置等于对尾rear的位置加1时,这个时候队列就是满的:
经过前面的构思,这个题目就很好理解了
但是还有一个问题很值得思考:
题目中对于循环队列的定义还有一个点很重要:
循环队列的一个好处是我们可以利用这个队列之前用过的空间。在一个普通队列里,一旦一个队列满了,我们就不能插入下一个元素,即使在队列前面仍有空间。但是使用循环队列,我们能使用这些空间去存储新的值。
什么意思呢?
也就是说,循环队列中我们如果在栈满了之后还想存储值,也是可以的,但是就要反复地使用之前用过的空间,会将其覆盖,所以尾指针rear和头指针front的位置的下标是会有覆盖的变化的
我们将循环队列形象地转换成数组:
这样你就能理解我上面所说的问题了!
你可以看到,队列为空时,按照题目的意思,front的位置时为rear+1的,在上图中,其实front的位置是0,rear的位置是3。
他们之间的关系就需要我们来求证一下了,因为在循环队列这个环形队列中,无论插入还是删除,都是从队头(或者是队尾)进行操作的!
我们其实就可以发现front的位置是与队列最大存储元素有关联的,上图中最大存储个数是3,当front存入4个元素时,存完第3个就满了,这个时候就应该重新从front位置开始存储,所以front(rear)和存储个数k有着以下关系:
就是说无论front的位置怎么移动,他最终都是在1-k的范围之内的
front = front % ( k + 1 )
现在,我们就可以开始用代码实现循环队列:
循环队列的构造
我们首先定义一个结构体,就是循环队列的结构
首先就是front和rear分别为队首和队尾的下标位置
然后就是k,存储元素个数
还有数组a,存储元素
typedef struct
{
int front;
int rear;
int k;
int* a;
} MyCircularQueue;
然后我们就可以构造一个循环队列了
MyCircularQueue* myCircularQueueCreate(int k)
{
MyCircularQueue* obj=(MyCircularQueue*)malloc(sizeof(MyCircularQueue));
obj->a=(int*)malloc(sizeof(int)*(k+1));
obj->front=obj->rear=0;
obj->k=k;
return obj;
}
判断循环队列是否为空
我们在前面的解题构思中就知道,当front和rear相等时,循环队列就为空了,所以我们直接返回obj->front==obj->rear,如果队列为空,就返回 1,队列不为空就返回0
bool myCircularQueueIsEmpty(MyCircularQueue* obj)
{
return obj->front==obj->rear;
}
判断循环队列是否已满
当rear+1和front相等时就是满的
这里能这样写吗?答案是不能,他要除以k+1然后取余,和front的方法一样
bool myCircularQueueIsFull(MyCircularQueue* obj)
{
return (obj->rear+1)%(obj->k+1)==obj->front;
}
循环队列插入元素
如果队列已经满了我们就直接返回false即可
如果不是满的话就要将数组rear位置下标的值赋值为你要插入的元素的值
同时rear++,然后取余,最后返回true
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value)
{
if(myCircularQueueIsFull(obj))
{
return false;
}
obj->a[obj->rear]=value;
obj->rear++;
obj->rear %= (obj->k+1);
return true;
}
循环队列删除元素
当队列为空时就不能删了,返回-1
不为空时,我们就将front的位置往前移动,这样队首的元素就被删除了
同时记得取余
bool myCircularQueueDeQueue(MyCircularQueue* obj)
{
if(myCircularQueueIsEmpty(obj))
{
return false;
}
obj->front++;
obj->front %= (obj->k+1);
return true;
}
获取循环队列队首元素
这个也很简单,直接返回数组的front下标位置的元素即可
int myCircularQueueFront(MyCircularQueue* obj)
{
if(myCircularQueueIsEmpty(obj))
{
return -1;
}
return obj->a[obj->front];
}
获取循环队列尾首元素
返回队尾元素我们就要根据图来具体求下标的关系了
由于画图较为麻烦,作者水平很有限,故不画图,给上源码,诸位大佬自己琢磨
int myCircularQueueRear(MyCircularQueue* obj)
{
if(myCircularQueueIsEmpty(obj))
{
return -1;
}
return obj->a[(obj->rear+obj->k)%(obj->k+1)];
}
循环队列的销毁
free循环队列目标的同时记得把数组也给free掉,不然可能会出现内存泄漏
void myCircularQueueFree(MyCircularQueue* obj)
{
free(obj->a);
free(obj);
}
完整代码如下:
typedef struct
{
int front;
int rear;
int k;
int* a;
} MyCircularQueue;
MyCircularQueue* myCircularQueueCreate(int k)
{
MyCircularQueue* obj=(MyCircularQueue*)malloc(sizeof(MyCircularQueue));
obj->a=(int*)malloc(sizeof(int)*(k+1));
obj->front=obj->rear=0;
obj->k=k;
return obj;
}
bool myCircularQueueIsEmpty(MyCircularQueue* obj)
{
return obj->front==obj->rear;
}
bool myCircularQueueIsFull(MyCircularQueue* obj)
{
return (obj->rear+1)%(obj->k+1)==obj->front;
}
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value)
{
if(myCircularQueueIsFull(obj))
{
return false;
}
obj->a[obj->rear]=value;
obj->rear++;
obj->rear %= (obj->k+1);
return true;
}
bool myCircularQueueDeQueue(MyCircularQueue* obj)
{
if(myCircularQueueIsEmpty(obj))
{
return false;
}
obj->front++;
obj->front %= (obj->k+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;
}
return obj->a[(obj->rear+obj->k)%(obj->k+1)];
}
void myCircularQueueFree(MyCircularQueue* obj)
{
free(obj->a);
free(obj);
}
好了,今天的分享到这里就结束了,谢谢大家的支持!