数据结构--循环队列
实现队列的方法很多,比如动态数组、链表,今天主要介绍循环队列
首先说用静态数组实现简单队列。
很显然,当队列满后,即便全部元素都出队,队列还是满的状态。这种情况就叫做“假溢出”,即数组中明明有可用空间,但却无法使用。
这是由定长数组的特性决定的。但我们可用改变一下思路,当队尾指针指向数组最后一个位置时,如果再有数据入队,并且队头指针没有指向数组的第一个元素,那么就让队为指针绕回到数组头部。这样就形成了一个逻辑上的环。
这样,只要队列中实际的元素数量小于数组长度减一,就可以继续入队了。
其实这是一个非常简单的数据结构,难点就是判断队空、对满,以及计算队列长度。
1
const int MAX_QUEUE_SIZE = 5;
2
template<typename T>
3
class cyc_queue
4
{
5
public:
6
cyc_queue()
7
:m_nHead(0),
8
m_nTail(0)
9
{}
10![]()
11![]()
12
//如对操作, 将数据追加到队列头部,并改变队首指针,如队成功,则返回true
13
bool in_queue(const T & data)
14
{
15
if(full())
16
//队列满
17
{
18
return false;
19
}
20![]()
21
m_array[m_nTail] = data;
22
m_nTail = (m_nTail + 1) % MAX_QUEUE_SIZE;
23
}
24![]()
25
//出队操作,将队首数据复制并返回,改变队首指针
26
T out_queue()
27
{
28
if(empty())
29
{
30
throw("队列已空");
31
}
32
33
T temp = m_array[m_nHead];
34
m_nHead = (m_nHead + 1) % MAX_QUEUE_SIZE;
35
return temp;
36
}
37![]()
38
bool empty()
39
{
40![]()
41
return m_nTail == m_nHead;
42
}
43![]()
44
bool full()
45
{
46
return (m_nTail + 1) % MAX_QUEUE_SIZE == m_nHead;
47
}
48![]()
49
size_t size()
50
{
51
return (m_nTail - m_nHead + MAX_QUEUE_SIZE) % MAX_QUEUE_SIZE;
52
}
53
private:
54
T m_array[MAX_QUEUE_SIZE];
55
int m_nHead;
56
int m_nTail;
57
};
const int MAX_QUEUE_SIZE = 5;2
template<typename T>3
class cyc_queue4
{5
public:6
cyc_queue()7
:m_nHead(0),8
m_nTail(0)9
{}10

11

12
//如对操作, 将数据追加到队列头部,并改变队首指针,如队成功,则返回true13
bool in_queue(const T & data)14
{15
if(full())16
//队列满17
{18
return false;19
}20

21
m_array[m_nTail] = data;22
m_nTail = (m_nTail + 1) % MAX_QUEUE_SIZE;23
}24

25
//出队操作,将队首数据复制并返回,改变队首指针26
T out_queue()27
{28
if(empty())29
{30
throw("队列已空");31
}32
33
T temp = m_array[m_nHead];34
m_nHead = (m_nHead + 1) % MAX_QUEUE_SIZE;35
return temp;36
}37

38
bool empty()39
{40

41
return m_nTail == m_nHead;42
}43

44
bool full()45
{46
return (m_nTail + 1) % MAX_QUEUE_SIZE == m_nHead;47
}48

49
size_t size()50
{51
return (m_nTail - m_nHead + MAX_QUEUE_SIZE) % MAX_QUEUE_SIZE;52
}53
private:54
T m_array[MAX_QUEUE_SIZE];55
int m_nHead;56
int m_nTail;57
};

