Arrays

Must-Know Array DSA Topics with Time & Space Complexity

Below is a comprehensive list of must-know Array DSA topics, including time & space complexity for common operations.


๐Ÿ”น Basic Array Operations

OperationTime ComplexitySpace ComplexityDescription
Access ElementO(1)O(1)Direct index access.
Update ElementO(1)O(1)Directly modifying an element at an index.
Insert at EndO(1) (Amortized)O(1)Only for dynamic arrays.
Insert at Beginning / MiddleO(n)O(1)Needs shifting of elements.
Delete at EndO(1)O(1)Removes last element.
Delete at Beginning / MiddleO(n)O(1)Needs shifting of elements.
Search (Unsorted)O(n)O(1)Linear search.
Search (Sorted)O(log n)O(1)Binary search (divide & conquer).
Sort ArrayO(n log n)O(1) or O(n)Merge Sort or Quick Sort.

๐Ÿ”น Substring, Subsequence & Related Patterns

Substring vs Subsequence vs Subarray vs Other Array Concepts

These terms are often confused in data structures & algorithms (DSA), so letโ€™s break them down with definitions, examples, and properties.


๐Ÿ”น 1. Substring

Definition:

A substring is a continuous part of a string. It must maintain contiguous characters in the same order.

Example:

For the string โ€œabcdeโ€, some substrings are:

  • "abc" โœ… (Valid)
  • "bcd" โœ… (Valid)
  • "ace" โŒ (Invalid, non-contiguous)

Properties:

  • Order matters โœ…
  • Contiguous characters โœ…
  • Number of substrings: ( O(n^2) ) (including single characters)
  • Operations: Can be extracted using s.substring(start, end)

Code (Java):

public class SubstringExample {
    public static void main(String[] args) {
        String str = "abcde";
        System.out.println(str.substring(1, 4)); // "bcd"
    }
}

๐Ÿ”น Brute Force:

  • Generate all substrings using nested loops.
  • Time Complexity: O(n2)O(n2)
  • Space Complexity: O(1)O(1)
javaCopyEdit// Brute Force - Generate all substrings
public class SubstringBruteForce {
    public static void main(String[] args) {
        String str = "abc";
        for (int i = 0; i < str.length(); i++) {
            for (int j = i + 1; j <= str.length(); j++) {
                System.out.println(str.substring(i, j));
            }
        }
    }
}

๐Ÿ”น Optimal Approach:

  • Sliding Window (For problems like Longest Unique Substring)
  • Time Complexity: O(n)O(n) (for some problems like Longest Unique Substring)
  • Space Complexity: O(1)O(1)

๐Ÿ”น 2. Subsequence

Definition:

A subsequence is a sequence derived by removing some or no characters from a string without changing their order.

Example:

For the string โ€œabcdeโ€, some subsequences are:

  • "abc" โœ… (Valid)
  • "ace" โœ… (Valid)
  • "edc" โŒ (Invalid, order changed)

Properties:

  • Order matters โœ…
  • Contiguous characters NOT required โŒ
  • Number of subsequences: ( 2^n ) (Each character can either be included or excluded)
  • Operations: Typically solved using recursion or dynamic programming (e.g., LCS โ€“ Longest Common Subsequence)

Code (Java โ€“ Generate Subsequences)

import java.util.ArrayList;
import java.util.List;

public class Subsequences {
    public static void generateSubsequences(String str, int index, String curr, List<String> result) {
        if (index == str.length()) {
            result.add(curr);
            return;
        }
        generateSubsequences(str, index + 1, curr + str.charAt(index), result); // Include
        generateSubsequences(str, index + 1, curr, result); // Exclude
    }

    public static void main(String[] args) {
        List<String> result = new ArrayList<>();
        generateSubsequences("abc", 0, "", result);
        System.out.println(result);
    }
}

Output: [abc, ab, ac, a, bc, b, c, ""] (Total: ( 2^3 = 8 ))

๐Ÿ”น Brute Force:

  • Generate all subsequences using recursion.
  • Time Complexity: O(2n)O(2n)
  • Space Complexity: O(n)O(n) (recursion stack)

๐Ÿ”น Optimal Approach:

Space Complexity: O(n)O(n)

Dynamic Programming (LIS, LCS, etc.)

Time Complexity: O(n2)O(n2) (for Longest Increasing Subsequence using DP)

๐Ÿ“Œ Common Problems:

  • Longest Increasing Subsequence (LIS)
  • Brute Force โ†’ O(2โฟ)
  • DP (O(nยฒ)), Binary Search (O(n log n))
  • Subset Sum / Combination Sum
  • Recursion โ†’ O(2โฟ)
  • DP โ†’ O(n * sum)
  • Longest Common Subsequence (LCS)
  • DP โ†’ O(n * m)

๐Ÿ”น 3. Subarray

Definition:

A subarray is a continuous part of an array.

Example:

For the array [1, 2, 3, 4], some subarrays are:

  • [1, 2] โœ… (Valid)
  • [2, 3, 4] โœ… (Valid)
  • [1, 3] โŒ (Invalid, non-contiguous)

Properties:

  • Order matters โœ…
  • Contiguous elements only โœ…
  • Number of subarrays: ( O(n^2) )
  • Common problems: Kadaneโ€™s Algorithm (Maximum Sum Subarray)

๐Ÿ”น Brute Force:

  • Generate all subarrays using nested loops.
  • Time Complexity: O(n2)O(n2)
  • Space Complexity: O(1)O(1)

Code (Java โ€“ Generate All Subarrays)

public class SubarrayExample {
    public static void main(String[] args) {
        int[] arr = {1, 2, 3, 4};
        for (int i = 0; i < arr.length; i++) {
            for (int j = i; j < arr.length; j++) {
                for (int k = i; k <= j; k++) {
                    System.out.print(arr[k] + " ");
                }
                System.out.println();
            }
        }
    }
}

Output (All Subarrays of [1,2,3,4]):

1 
1 2 
1 2 3 
1 2 3 4 
2 
2 3 
2 3 4 
3 
3 4 
4 

๐Ÿ”น Optimal Approach:

  • Kadaneโ€™s Algorithm (Maximum Sum Subarray)
  • Time Complexity: O(n)O(n)
  • Space Complexity: O(1)O(1)
javaCopyEditpublic class KadaneAlgorithm {
    public static int maxSubarraySum(int[] arr) {
        int maxSum = Integer.MIN_VALUE, currSum = 0;
        for (int num : arr) {
            currSum = Math.max(num, currSum + num);
            maxSum = Math.max(maxSum, currSum);
        }
        return maxSum;
    }

    public static void main(String[] args) {
        int[] arr = {-2, 1, -3, 4, -1, 2, 1, -5, 4};
        System.out.println(maxSubarraySum(arr)); // Output: 6
    }
}

Code (Java โ€“ Generate All Subarrays)

๐Ÿ“Œ Common Problems:

  • Kadaneโ€™s Algorithm (Max Sum Subarray) โ†’ O(n)
  • Prefix Sum (Range Sum Queries) โ†’ O(n) precompute, O(1) query
  • Sliding Window (Fixed/Variable Window) โ†’ O(n)

๐Ÿ”น 4. Subset

Definition:

A subset is any possible selection of elements from an array in any order.

Example:

For the array [1, 2, 3], some subsets are:

  • [] โœ… (Empty subset)
  • [1] โœ…
  • [2, 3] โœ…
  • [3, 1] โœ… (Order doesnโ€™t matter)
  • [1, 3, 2] โœ… (Also valid)

Properties:

  • Order does NOT matter โŒ
  • Contiguity NOT required โŒ
  • Number of subsets: ( 2^n ) (Same as subsequences)
  • Common problems: Subset Sum, Power Set

๐Ÿ”น Brute Force:

Space Complexity: O(n)O(n) (recursion stack)

Generate all subsets using recursion.

Time Complexity: O(2n)O(2n)

import java.util.*;

public class SubsetExample {
    public static void generateSubsets(int[] arr, int index, List<Integer> current, List<List<Integer>> result) {
        if (index == arr.length) {
            result.add(new ArrayList<>(current));
            return;
        }
        generateSubsets(arr, index + 1, current, result); // Exclude
        current.add(arr[index]);
        generateSubsets(arr, index + 1, current, result); // Include
        current.remove(current.size() - 1);
    }

    public static void main(String[] args) {
        int[] arr = {1, 2, 3};
        List<List<Integer>> result = new ArrayList<>();
        generateSubsets(arr, 0, new ArrayList<>(), result);
        System.out.println(result);
    }
}

Output: [[], [3], [2], [2, 3], [1], [1, 3], [1, 2], [1, 2, 3]]

๐Ÿ”น Optimal Approach:

  • Bit Manipulation (Power Set)
  • Time Complexity: O(2n)O(2n) (No way to improve beyond brute force)
  • Space Complexity: O(1)O(1) (If only printing subsets)
javaCopyEditimport java.util.*;

public class SubsetBitManipulation {
    public static void main(String[] args) {
        int[] arr = {1, 2, 3};
        int n = arr.length;
        List<List<Integer>> result = new ArrayList<>();

        for (int i = 0; i < (1 << n); i++) { // 2^n subsets
            List<Integer> subset = new ArrayList<>();
            for (int j = 0; j < n; j++) {
                if ((i & (1 << j)) != 0) { // Check if jth bit is set
                    subset.add(arr[j]);
                }
            }
            result.add(subset);
        }
        System.out.println(result);
    }
}

๐Ÿ”น Key Differences Table

ConceptContiguous Required?Order Matters?Total Count
Substringโœ… Yesโœ… Yes( O(n^2) )
SubsequenceโŒ Noโœ… Yes( 2^n )
Subarrayโœ… Yesโœ… Yes( O(n^2) )
SubsetโŒ NoโŒ No( 2^n )

๐Ÿ”น Practical Use Cases

Problem TypeCommon Approach
Find max sum subarrayKadaneโ€™s Algorithm
Longest increasing subsequenceDynamic Programming (LIS)
Find all subsets (power set)Backtracking/Bit Manipulation
Find all subarraysNested loops or sliding window
Check if one string is a subsequence of anotherTwo-pointer technique

โœ… Summary

  • Substring = Continuous part of a string (O(nยฒ))
  • Subsequence = Any order-preserved sequence (O(2โฟ))
  • Subarray = Continuous part of an array (O(nยฒ))
  • Subset = Any selection of elements (O(2โฟ))

๐Ÿ”น Patterns in Arrays

1๏ธโƒฃ Prefix Sum

๐Ÿ“Œ Pattern: Precompute cumulative sums to answer range sum queries in O(1) time.

๐Ÿ”น Use Cases:

  • Range Sum Query (O(n) precompute, O(1) query)
  • Product of Array Except Self (O(n) time, O(1) space)
  • Find equilibrium index (where left sum = right sum)

2๏ธโƒฃ Sliding Window (Fixed & Variable)

๐Ÿ“Œ Pattern: Maintain a window & expand/shrink based on conditions.

๐Ÿ”น Use Cases:

  • Fixed-size window (e.g., max sum of subarray of size k) โ†’ O(n)
  • Variable-size window (e.g., smallest subarray sum โ‰ฅ target) โ†’ O(n)

3๏ธโƒฃ Two Pointers

๐Ÿ“Œ Pattern: Use two indices to reduce search space efficiently.

๐Ÿ”น Use Cases:

  • Sorted Array Two Sum (O(n))
  • 3Sum / 4Sum (O(nยฒ))
  • Container with Most Water (O(n))

4๏ธโƒฃ Binary Search on Arrays

๐Ÿ“Œ Pattern: Search in a sorted array efficiently in O(log n).

๐Ÿ”น Use Cases:

  • Find first/last occurrence of element
  • Find peak element
  • Binary Search on Answers (e.g., Minimize Maximum Distance, Split Array)

๐Ÿ”น Must-Know Problems

ProblemApproachTime Complexity
Two SumHash MapO(n)
3SumSorting + Two PointersO(nยฒ)
Maximum Subarray (Kadaneโ€™s)DPO(n)
Longest Consecutive SequenceHashSetO(n)
Product of Array Except SelfPrefix/SuffixO(n)
Minimum Window SubstringSliding WindowO(n)
Trapping Rain WaterTwo PointersO(n)

Here are Java code examples for common array operations:

๐Ÿ”น Array Util Java Methods:

Arrays.sort(numbers);

int[] numbersCopy = Arrays.copyOf(numbers, numbers.length);

int[] rangeCopy = Arrays.copyOfRange(originalArray, 1, 4); // Copies elements from index 1 (inclusive) to 4 (exclusive)

java.util.List<Integer> numberList = Arrays.asList(5, 2, 8, 1, 9, 4);

boolean isEqual = Arrays.equals(numbers, numbers2);

int index = Arrays.binarySearch(numbers, 4);

Arrays.fill(numbers, 0);


๐Ÿ”น Basic Array Operations

1๏ธโƒฃ Access, Update, Insert & Delete

import java.util.Arrays;

public class ArrayOperations {
    public static void main(String[] args) {
        int[] arr = {10, 20, 30, 40, 50};

        // Access
        System.out.println("Element at index 2: " + arr[2]); // O(1)

        // Update
        arr[2] = 100; // O(1)
        System.out.println("Updated array: " + Arrays.toString(arr));

        // Insert at the end
        arr = Arrays.copyOf(arr, arr.length + 1); // O(1) amortized
        arr[arr.length - 1] = 60;
        System.out.println("Array after inserting 60: " + Arrays.toString(arr));

        // Insert at index 2 (shifting elements)
        int insertIndex = 2;
        int insertValue = 99;
        arr = Arrays.copyOf(arr, arr.length + 1);
        System.arraycopy(arr, insertIndex, arr, insertIndex + 1, arr.length - insertIndex - 1);
        arr[insertIndex] = insertValue; // O(n)
        System.out.println("Array after inserting 99 at index 2: " + Arrays.toString(arr));

        // Delete element at index 3
        int deleteIndex = 3;
        System.arraycopy(arr, deleteIndex + 1, arr, deleteIndex, arr.length - deleteIndex - 1);
        arr = Arrays.copyOf(arr, arr.length - 1); // O(n)
        System.out.println("Array after deleting element at index 3: " + Arrays.toString(arr));
    }
}

๐Ÿ”น Searching in an Array

1๏ธโƒฃ Linear Search (Unsorted)

public class LinearSearch {
    public static int search(int[] arr, int target) {
        for (int i = 0; i < arr.length; i++) {
            if (arr[i] == target) return i; // O(n)
        }
        return -1;
    }

    public static void main(String[] args) {
        int[] arr = {3, 7, 1, 9, 5};
        System.out.println("Element found at index: " + search(arr, 9));
    }
}

2๏ธโƒฃ Binary Search (Sorted)

import java.util.Arrays;

public class BinarySearch {
    public static int binarySearch(int[] arr, int target) {
        int left = 0, right = arr.length - 1;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (arr[mid] == target) return mid;
            else if (arr[mid] < target) left = mid + 1;
            else right = mid - 1;
        }
        return -1; // O(log n)
    }

    public static void main(String[] args) {
        int[] arr = {1, 3, 5, 7, 9};
        System.out.println("Element found at index: " + binarySearch(arr, 5));
    }
}

๐Ÿ”น Prefix Sum (Range Sum)

public class PrefixSum {
    public static int[] prefixSum(int[] arr) {
        int[] prefix = new int[arr.length];
        prefix[0] = arr[0];
        for (int i = 1; i < arr.length; i++) {
            prefix[i] = prefix[i - 1] + arr[i]; // O(n)
        }
        return prefix;
    }

    public static int rangeSum(int[] prefix, int left, int right) {
        if (left == 0) return prefix[right];
        return prefix[right] - prefix[left - 1]; // O(1)
    }

    public static void main(String[] args) {
        int[] arr = {2, 3, 5, 1, 6};
        int[] prefix = prefixSum(arr);
        System.out.println("Sum of range [1,3]: " + rangeSum(prefix, 1, 3));
    }
}

๐Ÿ”น Sliding Window (Max Sum Subarray of Fixed Size K)

public class SlidingWindow {
    public static int maxSumSubarray(int[] arr, int k) {
        int maxSum = 0, windowSum = 0;
        for (int i = 0; i < k; i++) windowSum += arr[i];

        maxSum = windowSum;
        for (int i = k; i < arr.length; i++) {
            windowSum += arr[i] - arr[i - k]; // O(n)
            maxSum = Math.max(maxSum, windowSum);
        }
        return maxSum;
    }

    public static void main(String[] args) {
        int[] arr = {1, 2, 3, 4, 5, 6, 7};
        System.out.println("Max sum of 3-size subarray: " + maxSumSubarray(arr, 3));
    }
}

๐Ÿ”น Two Pointer (Pair Sum)

import java.util.Arrays;

public class TwoPointer {
    public static boolean twoSum(int[] arr, int target) {
        Arrays.sort(arr); // O(n log n)
        int left = 0, right = arr.length - 1;
        while (left < right) {
            int sum = arr[left] + arr[right];
            if (sum == target) return true;
            else if (sum < target) left++;
            else right--;
        }
        return false; // O(n)
    }

    public static void main(String[] args) {
        int[] arr = {4, 1, 6, 3, 8};
        System.out.println("Pair exists: " + twoSum(arr, 9));
    }
}

๐Ÿ”น Kadaneโ€™s Algorithm (Max Subarray Sum)

public class Kadane {
    public static int maxSubarraySum(int[] arr) {
        int maxSum = arr[0], currentSum = arr[0];
        for (int i = 1; i < arr.length; i++) {
            currentSum = Math.max(arr[i], currentSum + arr[i]);
            maxSum = Math.max(maxSum, currentSum); // O(n)
        }
        return maxSum;
    }

    public static void main(String[] args) {
        int[] arr = {-2, 1, -3, 4, -1, 2, 1, -5, 4};
        System.out.println("Max Subarray Sum: " + maxSubarraySum(arr));
    }
}

๐Ÿ”น Product of Array Except Self (O(n) Time, O(1) Space)

import java.util.Arrays;

public class ProductExceptSelf {
    public static int[] productExceptSelf(int[] arr) {
        int n = arr.length;
        int[] result = new int[n];
        Arrays.fill(result, 1);

        int leftProduct = 1, rightProduct = 1;
        for (int i = 0; i < n; i++) {
            result[i] *= leftProduct;
            leftProduct *= arr[i]; // Prefix product
        }
        for (int i = n - 1; i >= 0; i--) {
            result[i] *= rightProduct;
            rightProduct *= arr[i]; // Suffix product
        }
        return result; // O(n) time, O(1) space
    }

    public static void main(String[] args) {
        int[] arr = {1, 2, 3, 4};
        System.out.println("Product Except Self: " + Arrays.toString(productExceptSelf(arr)));
    }
}

โœ… Summary

โœ… Basic Operations: Insert, Delete, Search
โœ… Sorting & Searching: Binary Search, Merge/Quick Sort
โœ… Patterns: Prefix Sum, Sliding Window, Two Pointers
โœ… Advanced Algorithms: Kadaneโ€™s Algorithm, Product Except Self

Leave a Reply