Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
package com.thealgorithms.sorts;

/**
* Recursive Insertion Sort algorithm.
*
* This is a recursive implementation of the standard Insertion Sort algorithm.
* Instead of iterating through the array, it sorts the first n-1 elements recursively
* and then inserts the nth element into its correct position.
*
* Concept:
* - Divide the problem into smaller subproblems by sorting first n-1 elements.
* - Insert the last element into the sorted portion.
*
* Time Complexity:
* - Best case: O(n) – array is already sorted
* - Average case: O(n^2)
* - Worst case: O(n^2) – array is reverse sorted
*
* Space Complexity:
* - O(n) – due to recursion stack
*
* Note:
* - This implementation is mainly useful for understanding recursion.
* - Iterative insertion sort is preferred in production due to lower space overhead.
*
* @see SortAlgorithm
*/
public class RecursiveInsertionSort implements SortAlgorithm {

/**
* Sorts the given array using recursive insertion sort.
*
* @param array The array to be sorted
* @param <T> The type of elements in the array, which must be comparable
* @return The sorted array
*/
@Override
public <T extends Comparable<T>> T[] sort(T[] array) {
if (array == null || array.length <= 1) {
return array;
}

recursiveSort(array, array.length);
return array;
}

/**
* Recursively sorts the first n elements of the array.
*
* @param array The array to be sorted
* @param n The number of elements to sort
* @param <T> The type of elements in the array
*/
private <T extends Comparable<T>> void recursiveSort(T[] array, int n) {

// Base case: single element is already sorted
if (n <= 1) {
return;
}

// Recursively sort first n-1 elements
recursiveSort(array, n - 1);

// Insert the nth element into the sorted subarray
insert(array, n);
}

/**
* Inserts the nth element into its correct position in the sorted subarray [0...n-2].
*
* @param array The array containing sorted subarray and one unsorted element
* @param n The size of the subarray to consider
* @param <T> The type of elements in the array
*/
private <T extends Comparable<T>> void insert(T[] array, int n) {
final T key = array[n - 1];
int j = n - 2;

// Shift elements greater than key to one position ahead
while (j >= 0 && SortUtils.less(key, array[j])) {
array[j + 1] = array[j];
j--;
}

// Place key at correct position
array[j + 1] = key;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
package com.thealgorithms.sorts;

import static org.junit.jupiter.api.Assertions.assertArrayEquals;

import org.junit.jupiter.api.Test;

class RecursiveInsertionSortTest {

private final RecursiveInsertionSort sorter = new RecursiveInsertionSort();

@Test
void testEmptyArray() {
Integer[] input = {};
Integer[] expected = {};
assertArrayEquals(expected, sorter.sort(input));
}

@Test
void testSingleElement() {
Integer[] input = {1};
Integer[] expected = {1};
assertArrayEquals(expected, sorter.sort(input));
}

@Test
void testAlreadySorted() {
Integer[] input = {1, 2, 3, 4, 5};
Integer[] expected = {1, 2, 3, 4, 5};
assertArrayEquals(expected, sorter.sort(input));
}

@Test
void testReverseSorted() {
Integer[] input = {5, 4, 3, 2, 1};
Integer[] expected = {1, 2, 3, 4, 5};
assertArrayEquals(expected, sorter.sort(input));
}

@Test
void testRandomOrder() {
Integer[] input = {3, 1, 4, 5, 2};
Integer[] expected = {1, 2, 3, 4, 5};
assertArrayEquals(expected, sorter.sort(input));
}

@Test
void testDuplicates() {
Integer[] input = {4, 2, 5, 2, 3};
Integer[] expected = {2, 2, 3, 4, 5};
assertArrayEquals(expected, sorter.sort(input));
}

@Test
void testStrings() {
String[] input = {"banana", "apple", "cherry"};
String[] expected = {"apple", "banana", "cherry"};
assertArrayEquals(expected, sorter.sort(input));
}
}
Loading