Skip to content

Commit 1feceb7

Browse files
authored
Add MLFQ Scheduler (#5575)
1 parent 4008e49 commit 1feceb7

File tree

3 files changed

+200
-0
lines changed

3 files changed

+200
-0
lines changed

DIRECTORY.md

+6
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@
4242
* [AES](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/ciphers/AES.java)
4343
* [AESEncryption](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/ciphers/AESEncryption.java)
4444
* [AffineCipher](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/ciphers/AffineCipher.java)
45+
* [Autokey](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/ciphers/Autokey.java)
4546
* [Blowfish](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/ciphers/Blowfish.java)
4647
* [Caesar](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/ciphers/Caesar.java)
4748
* [ColumnarTranspositionCipher](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/ciphers/ColumnarTranspositionCipher.java)
@@ -203,6 +204,7 @@
203204
* [SameTreesCheck](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/datastructures/trees/SameTreesCheck.java)
204205
* [SegmentTree](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/datastructures/trees/SegmentTree.java)
205206
* [SplayTree](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/datastructures/trees/SplayTree.java)
207+
* [Treap](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/datastructures/trees/Treap.java)
206208
* [TreeRandomNode](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/datastructures/trees/TreeRandomNode.java)
207209
* [TrieImp](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/datastructures/trees/TrieImp.java)
208210
* [VerticalOrderTraversal](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/datastructures/trees/VerticalOrderTraversal.java)
@@ -457,6 +459,7 @@
457459
* [GenerateSubsets](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/Recursion/GenerateSubsets.java)
458460
* scheduling
459461
* [FCFSScheduling](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/scheduling/FCFSScheduling.java)
462+
* [MLFQScheduler](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/scheduling/MLFQScheduler.java)
460463
* [PreemptivePriorityScheduling](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/scheduling/PreemptivePriorityScheduling.java)
461464
* [RRScheduling](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/scheduling/RRScheduling.java)
462465
* [SJFScheduling](https://github.com/TheAlgorithms/Java/blob/master/src/main/java/com/thealgorithms/scheduling/SJFScheduling.java)
@@ -618,6 +621,7 @@
618621
* ciphers
619622
* a5
620623
* [LFSRTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/ciphers/a5/LFSRTest.java)
624+
* [AutokeyTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/ciphers/AutokeyTest.java)
621625
* [BlowfishTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/ciphers/BlowfishTest.java)
622626
* [CaesarTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/ciphers/CaesarTest.java)
623627
* [DESTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/ciphers/DESTest.java)
@@ -728,6 +732,7 @@
728732
* [PreOrderTraversalTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/datastructures/trees/PreOrderTraversalTest.java)
729733
* [SameTreesCheckTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/datastructures/trees/SameTreesCheckTest.java)
730734
* [SplayTreeTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/datastructures/trees/SplayTreeTest.java)
735+
* [TreapTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/datastructures/trees/TreapTest.java)
731736
* [TreeTestUtils](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/datastructures/trees/TreeTestUtils.java)
732737
* [TrieImpTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/datastructures/trees/TrieImpTest.java)
733738
* [VerticalOrderTraversalTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/datastructures/trees/VerticalOrderTraversalTest.java)
@@ -911,6 +916,7 @@
911916
* [GenerateSubsetsTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/Recursion/GenerateSubsetsTest.java)
912917
* scheduling
913918
* [FCFSSchedulingTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/scheduling/FCFSSchedulingTest.java)
919+
* [MLFQSchedulerTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/scheduling/MLFQSchedulerTest.java)
914920
* [PreemptivePrioritySchedulingTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/scheduling/PreemptivePrioritySchedulingTest.java)
915921
* [RRSchedulingTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/scheduling/RRSchedulingTest.java)
916922
* [SJFSchedulingTest](https://github.com/TheAlgorithms/Java/blob/master/src/test/java/com/thealgorithms/scheduling/SJFSchedulingTest.java)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,148 @@
1+
package com.thealgorithms.scheduling;
2+
3+
import java.util.ArrayList;
4+
import java.util.LinkedList;
5+
import java.util.List;
6+
import java.util.Queue;
7+
8+
/**
9+
* The Multi-Level Feedback Queue (MLFQ) Scheduler class.
10+
* This class simulates scheduling using multiple queues, where processes move
11+
* between queues depending on their CPU burst behavior.
12+
*/
13+
public class MLFQScheduler {
14+
private List<Queue<Process>> queues; // Multi-level feedback queues
15+
private int[] timeQuantum; // Time quantum for each queue level
16+
private int currentTime; // Current time in the system
17+
18+
/**
19+
* Constructor to initialize the MLFQ scheduler with the specified number of
20+
* levels and their corresponding time quantums.
21+
*
22+
* @param levels Number of queues (priority levels)
23+
* @param timeQuantums Time quantum for each queue level
24+
*/
25+
public MLFQScheduler(int levels, int[] timeQuantums) {
26+
queues = new ArrayList<>(levels);
27+
for (int i = 0; i < levels; i++) {
28+
queues.add(new LinkedList<>());
29+
}
30+
timeQuantum = timeQuantums;
31+
currentTime = 0;
32+
}
33+
34+
/**
35+
* Adds a new process to the highest priority queue (queue 0).
36+
*
37+
* @param p The process to be added to the scheduler
38+
*/
39+
public void addProcess(Process p) {
40+
queues.get(0).add(p);
41+
}
42+
43+
/**
44+
* Executes the scheduling process by running the processes in all queues,
45+
* promoting or demoting them based on their completion status and behavior.
46+
* The process continues until all queues are empty.
47+
*/
48+
public void run() {
49+
while (!allQueuesEmpty()) {
50+
for (int i = 0; i < queues.size(); i++) {
51+
Queue<Process> queue = queues.get(i);
52+
if (!queue.isEmpty()) {
53+
Process p = queue.poll();
54+
int quantum = timeQuantum[i];
55+
56+
// Execute the process for the minimum of the time quantum or the remaining time
57+
int timeSlice = Math.min(quantum, p.remainingTime);
58+
p.execute(timeSlice);
59+
currentTime += timeSlice; // Update the system's current time
60+
61+
if (p.isFinished()) {
62+
System.out.println("Process " + p.pid + " finished at time " + currentTime);
63+
} else {
64+
if (i < queues.size() - 1) {
65+
p.priority++; // Demote the process to the next lower priority queue
66+
queues.get(i + 1).add(p); // Add to the next queue level
67+
} else {
68+
queue.add(p); // Stay in the same queue if it's the last level
69+
}
70+
}
71+
}
72+
}
73+
}
74+
}
75+
76+
/**
77+
* Helper function to check if all the queues are empty (i.e., no process is
78+
* left to execute).
79+
*
80+
* @return true if all queues are empty, otherwise false
81+
*/
82+
private boolean allQueuesEmpty() {
83+
for (Queue<Process> queue : queues) {
84+
if (!queue.isEmpty()) {
85+
return false;
86+
}
87+
}
88+
return true;
89+
}
90+
91+
/**
92+
* Retrieves the current time of the scheduler, which reflects the total time
93+
* elapsed during the execution of all processes.
94+
*
95+
* @return The current time in the system
96+
*/
97+
public int getCurrentTime() {
98+
return currentTime;
99+
}
100+
}
101+
102+
/**
103+
* Represents a process in the Multi-Level Feedback Queue (MLFQ) scheduling
104+
* algorithm.
105+
*/
106+
class Process {
107+
int pid;
108+
int burstTime;
109+
int remainingTime;
110+
int arrivalTime;
111+
int priority;
112+
113+
/**
114+
* Constructor to initialize a new process.
115+
*
116+
* @param pid Process ID
117+
* @param burstTime CPU Burst Time (time required for the process)
118+
* @param arrivalTime Arrival time of the process
119+
*/
120+
Process(int pid, int burstTime, int arrivalTime) {
121+
this.pid = pid;
122+
this.burstTime = burstTime;
123+
this.remainingTime = burstTime;
124+
this.arrivalTime = arrivalTime;
125+
this.priority = 0;
126+
}
127+
128+
/**
129+
* Executes the process for a given time slice.
130+
*
131+
* @param timeSlice The amount of time the process is executed
132+
*/
133+
public void execute(int timeSlice) {
134+
remainingTime -= timeSlice;
135+
if (remainingTime < 0) {
136+
remainingTime = 0;
137+
}
138+
}
139+
140+
/**
141+
* Checks if the process has finished execution.
142+
*
143+
* @return true if the process is finished, otherwise false
144+
*/
145+
public boolean isFinished() {
146+
return remainingTime == 0;
147+
}
148+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
package com.thealgorithms.scheduling;
2+
3+
import static org.junit.jupiter.api.Assertions.assertEquals;
4+
5+
import org.junit.jupiter.api.Test;
6+
7+
public class MLFQSchedulerTest {
8+
9+
@Test
10+
void testMLFQScheduling() {
11+
// Create MLFQ Scheduler with 3 levels and time quantum for each level
12+
int[] timeQuantums = {4, 8, 12}; // Example of different quantum for each queue
13+
MLFQScheduler scheduler = new MLFQScheduler(3, timeQuantums);
14+
15+
// Add processes to the scheduler
16+
scheduler.addProcess(new Process(1, 10, 0)); // pid=1, burstTime=10, arrivalTime=0
17+
scheduler.addProcess(new Process(2, 15, 0)); // pid=2, burstTime=15, arrivalTime=0
18+
scheduler.addProcess(new Process(3, 25, 0)); // pid=3, burstTime=25, arrivalTime=0
19+
20+
// Run the scheduler
21+
scheduler.run();
22+
23+
// Check current time after all processes are finished
24+
assertEquals(50, scheduler.getCurrentTime());
25+
}
26+
27+
@Test
28+
void testProcessCompletionOrder() {
29+
int[] timeQuantums = {3, 6, 9};
30+
MLFQScheduler scheduler = new MLFQScheduler(3, timeQuantums);
31+
32+
Process p1 = new Process(1, 10, 0);
33+
Process p2 = new Process(2, 5, 0);
34+
Process p3 = new Process(3, 20, 0);
35+
36+
scheduler.addProcess(p1);
37+
scheduler.addProcess(p2);
38+
scheduler.addProcess(p3);
39+
40+
scheduler.run();
41+
42+
// After running, current time should match the total burst time for all
43+
// processes
44+
assertEquals(35, scheduler.getCurrentTime());
45+
}
46+
}

0 commit comments

Comments
 (0)