mirror of
https://github.com/TheAlgorithms/Java.git
synced 2025-07-06 00:54:32 +08:00
Add Quick Select (#2860)
Co-authored-by: Nils Goldmann <goldman09@mi.fu-berlin.de> Co-authored-by: Andrii Siriak <siryaka@gmail.com>
This commit is contained in:
140
src/main/java/com/thealgorithms/searches/QuickSelect.java
Normal file
140
src/main/java/com/thealgorithms/searches/QuickSelect.java
Normal file
@ -0,0 +1,140 @@
|
||||
package com.thealgorithms.searches;
|
||||
|
||||
import java.util.*;
|
||||
|
||||
/**
|
||||
* An implementation of the Quickselect algorithm as described
|
||||
* <a href="https://en.wikipedia.org/wiki/Median_of_medians">here</a>.
|
||||
*/
|
||||
public final class QuickSelect {
|
||||
|
||||
/**
|
||||
* Selects the {@code n}-th largest element of {@code list}, i.e. the element that would
|
||||
* be at index n if the list was sorted.
|
||||
* <p>
|
||||
* Calling this function might change the order of elements in {@code list}.
|
||||
*
|
||||
* @param list the list of elements
|
||||
* @param n the index
|
||||
* @param <T> the type of list elements
|
||||
* @return the n-th largest element in the list
|
||||
* @throws IndexOutOfBoundsException if n is less than 0 or greater or equal to
|
||||
* the number of elements in the list
|
||||
* @throws IllegalArgumentException if the list is empty
|
||||
* @throws NullPointerException if {@code list} is null
|
||||
*/
|
||||
public static <T extends Comparable<T>> T select(List<T> list, int n) {
|
||||
Objects.requireNonNull(list, "The list of elements must not be null.");
|
||||
|
||||
if (list.size() == 0) {
|
||||
String msg = "The list of elements must not be empty.";
|
||||
throw new IllegalArgumentException(msg);
|
||||
}
|
||||
|
||||
if (n < 0) {
|
||||
String msg = "The index must not be negative.";
|
||||
throw new IndexOutOfBoundsException(msg);
|
||||
}
|
||||
|
||||
if (n >= list.size()) {
|
||||
String msg = "The index must be less than the number of elements.";
|
||||
throw new IndexOutOfBoundsException(msg);
|
||||
}
|
||||
|
||||
int index = selectIndex(list, n);
|
||||
return list.get(index);
|
||||
}
|
||||
|
||||
private static <T extends Comparable<T>> int selectIndex(List<T> list, int n) {
|
||||
return selectIndex(list, 0, list.size() - 1, n);
|
||||
}
|
||||
|
||||
private static <T extends Comparable<T>> int selectIndex(
|
||||
List<T> list,
|
||||
int left,
|
||||
int right,
|
||||
int n
|
||||
) {
|
||||
while (true) {
|
||||
if (left == right)
|
||||
return left;
|
||||
int pivotIndex = pivot(list, left, right);
|
||||
pivotIndex = partition(list, left, right, pivotIndex, n);
|
||||
if (n == pivotIndex) {
|
||||
return n;
|
||||
} else if (n < pivotIndex) {
|
||||
right = pivotIndex - 1;
|
||||
} else {
|
||||
left = pivotIndex + 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static <T extends Comparable<T>> int partition(
|
||||
List<T> list,
|
||||
int left,
|
||||
int right,
|
||||
int pivotIndex,
|
||||
int n
|
||||
) {
|
||||
T pivotValue = list.get(pivotIndex);
|
||||
Collections.swap(list, pivotIndex, right);
|
||||
int storeIndex = left;
|
||||
|
||||
for (int i = left; i < right; i++) {
|
||||
if (list.get(i).compareTo(pivotValue) < 0) {
|
||||
Collections.swap(list, storeIndex, i);
|
||||
storeIndex++;
|
||||
}
|
||||
}
|
||||
|
||||
int storeIndexEq = storeIndex;
|
||||
|
||||
for (int i = storeIndex; i < right; i++) {
|
||||
if (list.get(i).compareTo(pivotValue) == 0) {
|
||||
Collections.swap(list, storeIndexEq, i);
|
||||
storeIndexEq++;
|
||||
}
|
||||
}
|
||||
|
||||
Collections.swap(list, right, storeIndexEq);
|
||||
|
||||
return (n < storeIndex)
|
||||
? storeIndex
|
||||
: Math.min(n, storeIndexEq);
|
||||
}
|
||||
|
||||
private static <T extends Comparable<T>> int pivot(
|
||||
List<T> list,
|
||||
int left,
|
||||
int right
|
||||
) {
|
||||
if (right - left < 5) {
|
||||
return partition5(list, left, right);
|
||||
}
|
||||
|
||||
for (int i = left; i < right; i += 5) {
|
||||
int subRight = i + 4;
|
||||
if (subRight > right) {
|
||||
subRight = right;
|
||||
}
|
||||
int median5 = partition5(list, i, subRight);
|
||||
int rightIndex = left + (i - left) / 5;
|
||||
Collections.swap(list, median5, rightIndex);
|
||||
}
|
||||
|
||||
int mid = (right - left) / 10 + left + 1;
|
||||
int rightIndex = left + (right - left) / 5;
|
||||
return selectIndex(list, left, rightIndex, mid);
|
||||
}
|
||||
|
||||
private static <T extends Comparable<T>> int partition5(
|
||||
List<T> list,
|
||||
int left,
|
||||
int right
|
||||
) {
|
||||
List<T> ts = list.subList(left, right);
|
||||
ts.sort(Comparator.naturalOrder());
|
||||
return (left + right) >>> 1;
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user