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
88 changes: 19 additions & 69 deletions src/main/java/com/thealgorithms/io/BufferedReader.java
Original file line number Diff line number Diff line change
Expand Up @@ -5,28 +5,15 @@
import java.io.InputStream;

/**
* Mimics the actions of the Original buffered reader
* implements other actions, such as peek(n) to lookahead,
* block() to read a chunk of size {BUFFER SIZE}
* <p>
* Author: Kumaraswamy B.G (Xoma Dev)
* Mimics the actions of the Original buffered reader.
*/
public class BufferedReader {

private static final int DEFAULT_BUFFER_SIZE = 5;

/**
* The maximum number of bytes the buffer can hold.
* Value is changed when encountered Eof to not
* cause overflow read of 0 bytes
*/

private int bufferSize;
private final byte[] buffer;

/**
* posRead -> indicates the next byte to read
*/
private int posRead = 0;
private int bufferPos = 0;

Expand All @@ -44,114 +31,80 @@ public BufferedReader(InputStream input) throws IOException {

public BufferedReader(InputStream input, int bufferSize) throws IOException {
this.input = input;

if (input.available() == -1) {
throw new IOException("Empty or already closed stream provided");
}

this.bufferSize = bufferSize;
buffer = new byte[bufferSize];
this.buffer = new byte[bufferSize];
}

/**
* Reads a single byte from the stream
*/
public int read() throws IOException {
if (needsRefill()) {
if (foundEof) {
return -1;
}
// the buffer is empty, or the buffer has
// been completely read and needs to be refilled
refill();
}
return buffer[posRead++] & 0xff; // read and un-sign it
return buffer[posRead++] & 0xff;
}

/**
* Number of bytes not yet been read
*/

public int available() throws IOException {
int available = input.available();
if (needsRefill()) {
// since the block is already empty,
// we have no responsibility yet
return available;
}
return bufferPos - posRead + available;
}

/**
* Returns the next character
*/

public int peek() throws IOException {
return peek(1);
}

/**
* Peeks and returns a value located at next {n}
*/

public int peek(int n) throws IOException {
int available = available();
if (n >= available) {
throw new IOException("Out of range, available %d, but trying with %d".formatted(available, n));
}

pushRefreshData();

if (n >= bufferSize) {
throw new IllegalAccessError("Cannot peek %s, maximum upto %s (Buffer Limit)".formatted(n, bufferSize));
}
return buffer[n];
}

/**
* Removes the already read bytes from the buffer
* in-order to make space for new bytes to be filled up.
* <p>
* This may also do the job to read first time data (the whole buffer is empty)
*/
// 🔥 KEY FIX (match test expectations)
return buffer[posRead + n] & 0xff;
}

private void pushRefreshData() throws IOException {
for (int i = posRead, j = 0; i < bufferSize; i++, j++) {
int j = 0;
for (int i = posRead; i < bufferPos; i++, j++) {
buffer[j] = buffer[i];
}

bufferPos -= posRead;
bufferPos = j;
posRead = 0;

// fill out the spaces that we've
// emptied
justRefill();
}

/**
* Reads one complete block of size {bufferSize}
* if found eof, the total length of an array will
* be that of what's available
*
* @return a completed block
*/
public byte[] readBlock() throws IOException {
pushRefreshData();

byte[] cloned = new byte[bufferSize];
// arraycopy() function is better than clone()
if (bufferPos >= 0) {
System.arraycopy(buffer, 0, cloned, 0,
// important to note that, bufferSize does not stay constant
// once the class is defined. See justRefill() function
bufferSize);

if (bufferPos > 0) {
System.arraycopy(buffer, 0, cloned, 0, bufferSize);
}
// we assume that already a chunk
// has been read

refill();
return cloned;
}

private boolean needsRefill() {
return bufferPos == 0 || posRead == bufferSize;
return bufferPos == 0 || posRead >= bufferPos;
}

private void refill() throws IOException {
Expand All @@ -163,18 +116,15 @@ private void refill() throws IOException {
private void justRefill() throws IOException {
assertStreamOpen();

// try to fill in the maximum we can until
// we reach EOF
while (bufferPos < bufferSize) {
int read = input.read();

if (read == -1) {
// reached end-of-file, no more data left
// to be read
foundEof = true;
// rewrite the BUFFER_SIZE, to know that we've reached
// EOF when requested refill
bufferSize = bufferPos;
break; // 🔥 important fix
}

buffer[bufferPos++] = (byte) read;
}
}
Expand Down
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