顺序表实现队列

代码部分

cpp 复制代码
#include<iostream>
#include<stdexcept>
using namespace std;
template<typename T>
class Queue {
private:
	T* data;
	//int size;没有这个参数
	int front;
	int rear;
	int capacity;
	void resize();
public:
	Queue():data(new T[10]),front(0),rear(0),capacity(10){}
	void enqueue(T x);
	T dequeue();
	T getFront() const;//!
	int getSize() const;//!
	~Queue();
};
template<typename T>
void Queue<T>::resize() {
	T* newData = new T[capacity * 2];
	for (int i = front; i < rear; i++) {
		newData[i] = data[i];
	}
	delete[] data;
	data = newData;
	capacity *= 2;
}
template<typename T>
Queue<T>::~Queue() {
	delete[] data;
}
template<typename T>
void Queue<T>::enqueue(T x) {
	if (rear-front == capacity) {
		resize();
	}
	data[rear] = x;//!
	rear++;
}
template<typename T>
T Queue<T>::dequeue() {
	if (front == rear) {
		throw underflow_error("Queue is empty!");
	}
	front++;
	return data[front-1];//!
}
template<typename T>
T Queue<T>::getFront() const {
	if (front == rear) {
		throw underflow_error("Queue is empty!");
	}
	return data[front];
}
template<typename T>
int Queue<T>::getSize() const {
	return rear - front;
}
int main() {
	Queue<int> q;
	q.enqueue(1);
	q.enqueue(2);
	q.enqueue(3);
	q.enqueue(4);
	cout << q.dequeue()<<endl;
	cout << q.getSize()<<endl;
	return 0;
}

1.类的声明

cpp 复制代码
class Queue {
private:
	T* data;
	//int size;没有这个参数
	int front;
	int rear;
	int capacity;
	void resize();
public:
	Queue():data(new T[10]),front(0),rear(0),capacity(10){}
	void enqueue(T x);
	T dequeue();
	T getFront() const;//!
	int getSize() const;//!
	~Queue();
};

首先是队列的类的声明部分,咱们这个类名为Queue,即为队列,然后是私有权限的成员变量和函数,包括存放队列的数组,指向队列首位的指针front,指向队列最大索引位加一位置的指针rear,数组的容量capacity,以及扩容函数resize,然后是公共权限部分,包含用初始化列表初始各个成员变量的构造函数,以及入队,出队,获取队首元素、获取队列长度和析构。

2.具体函数实现

cpp 复制代码
template<typename T>
void Queue<T>::resize() {
	T* newData = new T[capacity * 2];
	for (int i = front; i < rear; i++) {
		newData[i] = data[i];
	}
	delete[] data;
	data = newData;
	capacity *= 2;
}
template<typename T>
Queue<T>::~Queue() {
	delete[] data;
}
template<typename T>
void Queue<T>::enqueue(T x) {
	if (rear-front == capacity) {
		resize();
	}
	data[rear] = x;//!
	rear++;
}
template<typename T>
T Queue<T>::dequeue() {
	if (front == rear) {
		throw underflow_error("Queue is empty!");
	}
	front++;
	return data[front-1];//!
}
template<typename T>
T Queue<T>::getFront() const {
	if (front == rear) {
		throw underflow_error("Queue is empty!");
	}
	return data[front];
}
template<typename T>
int Queue<T>::getSize() const {
	return rear - front;
}

具体函数实现部分,首先是扩容函数,经过模板声明和函数声明,与顺序表扩容原理一样,动态申请一个新的数组,把旧数组中的值传给新数组,然后删除旧数组对应的内存,然后让指针指向新的数组,同时容量更新。

cpp 复制代码
template<typename T>
void Queue<T>::resize() {
	T* newData = new T[capacity * 2];
	for (int i = front; i < rear; i++) {
		newData[i] = data[i];
	}
	delete[] data;
	data = newData;
	capacity *= 2;
}

然后是析构函数,就是将顺序表对应的内存释放即可。

cpp 复制代码
template<typename T>
Queue<T>::~Queue() {
	delete[] data;
}

然后是入队函数,经过模板声明和函数声明,函数声明部分,返回值类型为void,函数名,enqueue代表入队,然后参数列表为一个T类型的变量,代表待传入的元素,然后函数内部,首先判断数组是否还有空间,如果没有则调用扩容函数,然后在rear索引位即队列尾部插入元素,然后让rear指针向后偏移1位。

cpp 复制代码
template<typename T>
void Queue<T>::enqueue(T x) {
	if (rear-front == capacity) {
		resize();
	}
	data[rear] = x;//!
	rear++;
}

然后是出队函数,经过模板声明和函数声明,函数声明部分,返回值类型为T,函数名dequeue,即为出队,函数内部,首先判断队列是否为空,若为空则抛出异常,否则首先让头指针向后偏移一个位置,然后返回队列最前面的元素,即front-1位置的元素。

cpp 复制代码
template<typename T>
T Queue<T>::dequeue() {
	if (front == rear) {
		throw underflow_error("Queue is empty!");
	}
	front++;
	return data[front-1];//!
}

然后是获取队首元素的函数,经过模板声明和函数声明,返回值类型为T,由于队首元素为T类型,然后函数名后面加上括号,括号后面加上const,因为这个函数无需改变队列原本的值。函数实现里面,首先判断队列是否为空,若为空则抛出异常,否则返回front索引位的元素。

cpp 复制代码
template<typename T>
T Queue<T>::getFront() const {
	if (front == rear) {
		throw underflow_error("Queue is empty!");
	}
	return data[front];
}

然后是获取队列长度的函数,

cpp 复制代码
template<typename T>
int Queue<T>::getSize() const {
	return rear - front;
}

这里面跟获取队首元素一样,要加const,然后返回值类型为int,由于长度是int类型的,然后返回尾指针减去头指针,代表队列的长度。

反思

1.首先是获取队列长度和队首元素的函数,函数声明时,小括号要紧跟在函数名后面,不要写在const后面。

2.其次队列的实现没有size.

相关推荐
2401_892070981 天前
【Linux C++ 日志系统实战】LogFile 日志文件管理核心:滚动策略、线程安全与方法全解析
linux·c++·日志系统·日志滚动
yuzhuanhei1 天前
Visual Studio 配置C++opencv
c++·学习·visual studio
小O的算法实验室1 天前
2026年ASOC,基于深度强化学习的无人机三维复杂环境分层自适应导航规划方法,深度解析+性能实测
算法·无人机·论文复现·智能算法·智能算法改进
‎ദ്ദിᵔ.˛.ᵔ₎1 天前
LIST 的相关知识
数据结构·list
不爱吃炸鸡柳1 天前
C++ STL list 超详细解析:从接口使用到模拟实现
开发语言·c++·list
M--Y1 天前
Redis常用数据类型
数据结构·数据库·redis
十五年专注C++开发1 天前
RTTR: 一款MIT 协议开源的 C++ 运行时反射库
开发语言·c++·反射
‎ദ്ദിᵔ.˛.ᵔ₎1 天前
STL 栈 队列
开发语言·c++
2401_892070981 天前
【Linux C++ 日志系统实战】高性能文件写入 AppendFile 核心方法解析
linux·c++·日志系统·文件写对象
郭涤生1 天前
STL vector 扩容机制与自定义内存分配器设计分析
c++·算法