Let's implement a circular queue in Java. A circular queue, also known as a ring buffer, is a fixed-size data structure that operates as if the end is connected to the beginning. This is particularly useful in scenarios where you want to maintain a history of the most recent 'n' elements, such as in real-time data processing or managing tasks in an operating system. Your implementation should include the following functionalities:
For example, if the queue has a capacity of 5, and you enqueue elements 1, 2, 3, 4, and 5, the queue is full. If you then dequeue one element, the element 1 is removed, and you can enqueue a new element, say 6. The queue now contains 2, 3, 4, 5, and 6. Make sure your implementation handles the circular nature correctly, meaning the rear pointer should wrap around to the beginning of the array when it reaches the end, and the front pointer should wrap around similarly when dequeuing.
Provide a complete, runnable Java code solution with comments to explain the logic behind each method. Also include a simple main
method demonstrating the usage of your circular queue with some enqueue and dequeue operations.
Let's implement a circular queue in Java. A circular queue, also known as a ring buffer, is a fixed-size data structure that operates as if the end is connected to the beginning. This is particularly useful in scenarios where you want to maintain a history of the most recent 'n' elements, such as in real-time data processing or managing tasks in an operating system.
First, here's a straightforward implementation of a circular queue in Java. This will serve as our base, which we can then optimize.
public class CircularQueue {
private int[] queue;
private int head;
private int tail;
private int size;
private int capacity;
public CircularQueue(int capacity) {
this.capacity = capacity;
this.queue = new int[capacity];
this.head = 0;
this.tail = 0;
this.size = 0;
}
public boolean enqueue(int data) {
if (isFull()) {
System.out.println("Queue is full. Cannot enqueue.");
return false;
}
queue[tail] = data;
tail = (tail + 1) % capacity;
size++;
return true;
}
public Integer dequeue() {
if (isEmpty()) {
System.out.println("Queue is empty. Cannot dequeue.");
return null;
}
int data = queue[head];
head = (head + 1) % capacity;
size--;
return data;
}
public Integer peek() {
if (isEmpty()) {
return null;
}
return queue[head];
}
public boolean isEmpty() {
return size == 0;
}
public boolean isFull() {
return size == capacity;
}
public int size() {
return size;
}
public void printQueue() {
if (isEmpty()) {
System.out.println("Queue is empty.");
return;
}
System.out.print("Queue: ");
int i = head;
int count = 0;
while (count < size) {
System.out.print(queue[i] + " ");
i = (i + 1) % capacity;
count++;
}
System.out.println();
}
public static void main(String[] args) {
CircularQueue cq = new CircularQueue(5);
cq.enqueue(1);
cq.enqueue(2);
cq.enqueue(3);
cq.printQueue(); // Output: Queue: 1 2 3
cq.dequeue();
cq.printQueue(); // Output: Queue: 2 3
cq.enqueue(4);
cq.enqueue(5);
cq.enqueue(6);
cq.printQueue(); // Output: Queue: 2 3 4 5 6
cq.enqueue(7); // Output: Queue is full. Cannot enqueue.
cq.printQueue(); // Output: Queue: 2 3 4 5 6
cq.dequeue();
cq.enqueue(7);
cq.printQueue(); // Output: Queue: 3 4 5 6 7
}
}
The naive implementation is already quite efficient for basic operations. However, a slight optimization can be introduced to make the isFull
check more efficient by tracking the number of elements directly, as shown in the initial implementation. There isn't a drastically different approach that would yield significantly better performance due to the fundamental nature of the circular queue.
public class CircularQueueOptimized {
private int[] queue;
private int head;
private int tail;
private int size;
private int capacity;
public CircularQueueOptimized(int capacity) {
this.capacity = capacity;
this.queue = new int[capacity];
this.head = 0;
this.tail = 0;
this.size = 0;
}
public boolean enqueue(int data) {
if (isFull()) {
System.out.println("Queue is full. Cannot enqueue.");
return false;
}
queue[tail] = data;
tail = (tail + 1) % capacity;
size++;
return true;
}
public Integer dequeue() {
if (isEmpty()) {
System.out.println("Queue is empty. Cannot dequeue.");
return null;
}
int data = queue[head];
head = (head + 1) % capacity;
size--;
return data;
}
public Integer peek() {
if (isEmpty()) {
return null;
}
return queue[head];
}
public boolean isEmpty() {
return size == 0;
}
public boolean isFull() {
return size == capacity;
}
public int size() {
return size;
}
public void printQueue() {
if (isEmpty()) {
System.out.println("Queue is empty.");
return;
}
System.out.print("Queue: ");
int i = head;
int count = 0;
while (count < size) {
System.out.print(queue[i] + " ");
i = (i + 1) % capacity;
count++;
}
System.out.println();
}
public static void main(String[] args) {
CircularQueueOptimized cq = new CircularQueueOptimized(5);
cq.enqueue(1);
cq.enqueue(2);
cq.enqueue(3);
cq.printQueue();
cq.dequeue();
cq.printQueue();
cq.enqueue(4);
cq.enqueue(5);
cq.enqueue(6);
cq.printQueue();
cq.enqueue(7);
cq.printQueue();
cq.dequeue();
cq.enqueue(7);
cq.printQueue();
}
}
Empty Queue:
dequeue
or peek
from an empty queue.dequeue
and peek
should check if the queue is empty and return null
or an appropriate error message.Full Queue:
enqueue
into a full queue.enqueue
method should check if the queue is full and return false
or an error message.Queue Overflow:
isFull
check.Queue Underflow:
dequeue
more elements than available.isEmpty
check.Zero Capacity:
IllegalArgumentException
if the capacity is zero or negative.Integer Overflow/Underflow:
head
or tail
might overflow if the queue is used for a very long time.% capacity
inherently handles the wrap-around, preventing integer overflow issues.Null Elements: