I am going to have two functions. The first one is to handle the recursive splitting of the array into smaller parts, and the second one will handle merging those parts back together in sorted order.

The first function is the main merge_sort function. Its job is to recursively divide the input array into smaller parts until each part contains only one element or no elements.

The second function is called merge, and its purpose is to take two sorted arrays and combine them into a single sorted array

def merge_sort(arr):
    if len(arr) <= 1:
        return arr  # Base case: array with 0 or 1 elements is already sorted

    # Split the array into two halves
    mid = len(arr) // 2
    left = merge_sort(arr[:mid])
    right = merge_sort(arr[mid:])

    # Merge the sorted halves
    return merge(left, right)

def merge(left, right):
    sorted_array = []
    i = j = 0

    # Compare elements from left and right and merge them in sorted order
    while i < len(left) and j < len(right):
        if left[i] <= right[j]:
            sorted_array.append(left[i])
            i += 1
        else:
            sorted_array.append(right[j])
            j += 1

    # Add any remaining elements from left or right
    sorted_array.extend(left[i:])
    sorted_array.extend(right[j:])

    return sorted_array

# Example Usage
arr = [38, 27, 43, 3, 9, 82, 10]
sorted_arr = merge_sort(arr)
print("Sorted Array:", sorted_arr)
Time Complexity.
Best Case, Worst Case, and Average Case: O(nlogn)
Divide Step:

    The array is repeatedly divided into two halves until we reach arrays of size 1.
    The number of divisions is equal to the height of the recursion tree, which is log_2(n).
Merge Step:
    Each level of the recursion tree processes 𝑛 elements during merging.
    At each level, the merging of two halves requires O(n) time.

Total Work:
The total work across all levels of the recursion tree is:

O(n)+O(n)+... + O(n) (for log2(n) levels) =  O(nlogn)

Key Insight: The log 𝑛 factor comes from the depth of the recursion (splitting), and the 𝑛 factor comes from merging at each level.

Space Complexity.
O(n)
Why?

The algorithm requires additional space for the temporary arrays used during the merge step.
At any given time, we store a portion of the array in temporary arrays for merging, which can take up to O(n) space

Recursion Stack Space: 𝑂(log𝑛)
The recursion depth corresponds to the height of the recursion tree, which is log2(n)

total = log2(n) + O(n) = O(n)

21. Merge Two Sorted Lists

def mergeTwoLists(self, list1: Optional[ListNode], list2: Optional[ListNode]) -> Optional[ListNode]:
    dummy = ListNode()
    cur = dummy

    while list1 and list2:
        if list1.val < list2.val:
            cur.next = list1
            list1 = list1.next
        else:
            cur.next = list2
            list2 = list2.next
        cur = cur.next
    if list1:
        cur.next = list1
    if list2:
        cur.next = list2
    return dummy.next

# O(m+n)
# O(1)

23. Merge k Sorted Lists

class Solution:
    def mergeKLists(self, lists: List[Optional[ListNode]]) -> Optional[ListNode]:
        '''
        1. implement mergeTwoLists
        2. will divide and conquar method with mergeTwoLists
        '''
        if not lists:
            return None

        if len(lists) == 1:
            return lists[0]

        mid = len(lists) // 2 
        left = self.mergeKLists(lists[:mid])
        right = self.mergeKLists(lists[mid:])

        return self.mergeTwoLists(left, right)

    def mergeTwoLists(self, list1, list2):
        dummy = ListNode()
        cur = dummy

        while list1 and list2:
            if list1.val < list2.val:
                cur.next = list1
                list1 = list1.next
            else:
                cur.next = list2
                list2 = list2.next
            cur = cur.next

        if list1:
            cur.next = list1
        elif list2:
            cur.next = list2
        return dummy.next

# N: Total number of nodes across all linked lists.
# k: Number of linked lists.
#The algorithm splits the 𝑘 lists into two halves repeatedly until only one list remains.
# This takes log 𝑘 levels of merging.        

# At each level of merging, all 𝑁 nodes are processed across all pairs of lists.

#Space

# The algorithm divides the list recursively into two halves.
# At each recursive call, the depth of the recursion increases by 1.
# For 𝑘 lists, the depth of recursion is log 𝑘.

https://leetcode.com/problems/top-k-frequent-elements/description/?envType=company&envId=facebook&favoriteSlug=facebook-thirty-days

 

Here’s an updated structured script that includes the Quickselect approach for solving the Top K Frequent Elements problem, along with sorting, heap, and bucket sort methods.


You:
"To solve the problem of finding the k most frequent elements in a list of integers, there are a few different approaches we can take, depending on the input size and the desired level of efficiency. I’ll walk through each approach, from simplest to most optimized, including their pros and cons.

1. Sorting Solution

The simplest approach is to use sorting:

  • Steps: First, count the frequency of each element using a dictionary or Python's Counter class. Then, convert the frequency dictionary to a list of (element, frequency) tuples and sort this list by frequency in descending order. Finally, select the first k elements from the sorted list.
  • Time Complexity: O(n log n), due to sorting the entire list of elements by frequency.
  • Space Complexity: O(n) for the frequency dictionary and sorted list.
  • Pros:
    • Straightforward and easy to implement.
    • Suitable for small to moderate input sizes.
  • Cons:
    • Sorting is inefficient for large lists when we only need the top k elements. Sorting all elements doesn’t leverage the partial results we need.

2. Heap Solution (Efficient for Larger Lists with Small k)

A more efficient approach for larger inputs is to use a min-heap:

  • Steps: After creating a frequency dictionary, we use a min-heap to keep track of the k most frequent elements. For each element and its frequency, we push it into the heap. If the heap size exceeds k, we remove the element with the smallest frequency, ensuring that only the k most frequent elements remain.
  • Time Complexity: O(n log k), where we add n elements to a heap of size k.
  • Space Complexity: O(n) for the dictionary and O(k) for the heap.
  • Pros:
    • Efficient for large inputs where k is small relative to n.
    • Uses less space by storing only k elements in the heap.
  • Cons:
    • More complex to implement due to heap operations.

3. Bucket Sort Solution (Optimal for Frequency-Based Grouping)

An even more efficient approach in terms of time complexity is bucket sort:

  • Steps: After building the frequency dictionary, we create an array of buckets where each index represents a frequency count. Each bucket stores elements that appear that many times. Finally, we collect the top k elements by iterating through the buckets from highest to lowest frequency.
  • Time Complexity: O(n), as we only count elements and place them into frequency-based buckets.
  • Space Complexity: O(n) for the dictionary and buckets.
  • Pros:
    • Highly efficient for large inputs and avoids sorting or heap maintenance.
    • Works well for situations where k is close to n.
  • Cons:
    • Bucket sort can be less intuitive to implement, and requires extra space for the buckets.

4. Quickselect Solution (Optimal for Top-k Selection)

Another highly efficient solution, especially for very large lists, is Quickselect:

  • Steps: Quickselect is a partition-based algorithm similar to quicksort. After building the frequency dictionary, we convert it into a list of (element, frequency) pairs and use Quickselect to partially sort the list such that the k most frequent elements are positioned in the first k spots. We partition the list until the k-th most frequent element is in the correct position, and return the first k elements.
  • Time Complexity: O(n) on average, as Quickselect only partially sorts the list to find the top k elements.
  • Space Complexity: O(n) for the dictionary and list.
  • Pros:
    • Very efficient with an average-case complexity of O(n), especially for very large lists.
    • Avoids sorting the entire list, which makes it faster than the sorting approach.
  • Cons:
    • The worst-case complexity is O(n^2), though using random pivot selection helps mitigate this risk.
    • Quickselect is more complex to implement and understand compared to other solutions.

Summary:

  • Sorting: Simple but less efficient for large inputs, with O(n log n) complexity.
  • Heap: Ideal for large lists when k is much smaller than n, with O(n log k) complexity.
  • Bucket Sort: Optimized for large lists and frequency-based grouping, with O(n) complexity, though it requires additional space.
  • Quickselect: Offers the best average-case efficiency with O(n) time complexity, ideal for very large lists and when k is close to n.

Each solution has its trade-offs, so I’d choose the approach based on input size and constraints. For large lists with small k, the heap or Quickselect approach would be optimal, while for lists where k is close to n, bucket sort may be best."


This script provides a structured breakdown of each solution, explaining when each approach is optimal based on the constraints, making it easy to decide the best solution.


Problem Statement Recap:
You are given a list of integers and an integer k. The goal is to return the k most frequent elements in the list.


1. Sorting Solution

Explanation:

  • Start by counting the frequency of each element in the list. We can use Python's Counter from the collections module to achieve this.
  • Once we have the frequency of each element, we convert the frequency dictionary into a list of tuples, where each tuple is (element, frequency).
  • Sort this list of tuples in descending order based on frequency.
  • Finally, select the first k elements from this sorted list.

Implementation:

from collections import Counter
from typing import List

def top_k_frequent_sort(nums: List[int], k: int) -> List[int]:
    # Step 1: Count frequencies
    freq_count = Counter(nums)
    # Step 2: Sort items by frequency in descending order
    sorted_items = sorted(freq_count.items(), key=lambda item: item[1], reverse=True)
    # Step 3: Extract the first k elements
    return [item[0] for item in sorted_items[:k]]

# Example usage
nums = [1, 1, 1, 2, 2, 3]
k = 2
print(top_k_frequent_sort(nums, k))  # Output: [1, 2]

Time Complexity: O(n log n)

  • Counting frequencies takes O(n), while sorting the items by frequency takes O(n log n).

Space Complexity: O(n) for storing the frequency dictionary and sorted list.

Pros:

  • Simple and straightforward to implement.
  • Effective for small to medium inputs.

Cons:

  • Sorting the entire frequency list is unnecessary when we only need the top k elements, making it less efficient for large inputs.

2. Heap Solution (Optimal for Large Lists with Small k)

Explanation:

  • After counting the frequency of each element, we use a min-heap of size k to keep track of the k most frequent elements.
  • We push each element along with its frequency into the heap.
    • If the heap exceeds size k, we remove the element with the smallest frequency (root of the min-heap).
  • By the end, the heap contains only the k most frequent elements.

Implementation:

import heapq
from collections import Counter
from typing import List

def top_k_frequent_heap(nums: List[int], k: int) -> List[int]:
    # Step 1: Count frequencies
    freq_count = Counter(nums)
    # Step 2: Use a min-heap of size k
    min_heap = []
    for num, freq in freq_count.items():
        heapq.heappush(min_heap, (freq, num))
        if len(min_heap) > k:
            heapq.heappop(min_heap)
    # Step 3: Extract elements from the heap
    return [num for (freq, num) in min_heap]

# Example usage
nums = [1, 1, 1, 2, 2, 3]
k = 2
print(top_k_frequent_heap(nums, k))  # Output: [2, 1]

Time Complexity: O(n log k)

  • Counting frequencies takes O(n), and maintaining a heap of size k takes O(log k) time for each of the n elements.

Space Complexity: O(n) for the frequency dictionary and O(k) for the heap.

Pros:

  • Efficient for large inputs when k is much smaller than n.
  • Keeps track of only k elements, optimizing space and time usage.

Cons:

  • Slightly more complex to implement due to heap management.

3. Bucket Sort Solution (Optimal for Frequency-Based Grouping)

Explanation:

  • This approach leverages the fact that the frequency of elements is bounded by the length of the list (n), as the maximum frequency an element can have is n.
  • Create an array of n+1 empty buckets (index 0 to n). Each bucket at index i holds a list of elements that appear i times.
  • Place each element from the frequency dictionary into its corresponding bucket based on frequency.
  • Finally, iterate through the buckets in reverse order, collecting elements until we have k elements.

Implementation:

from collections import Counter
from typing import List

def top_k_frequent_bucket_sort(nums: List[int], k: int) -> List[int]:
    # Step 1: Count frequencies
    freq_count = Counter(nums)
    # Step 2: Initialize buckets where index is frequency
    buckets = [[] for _ in range(len(nums) + 1)]
    for num, freq in freq_count.items():
        buckets[freq].append(num)
    # Step 3: Gather top k elements from the buckets
    result = []
    for i in range(len(buckets) - 1, 0, -1):
        for num in buckets[i]:
            result.append(num)
            if len(result) == k:
                return result

# Example usage
nums = [1, 1, 1, 2, 2, 3]
k = 2
print(top_k_frequent_bucket_sort(nums, k))  # Output: [1, 2]

Time Complexity: O(n)

  • Counting frequencies takes O(n), and placing elements into buckets also takes O(n).

Space Complexity: O(n) for the frequency dictionary and buckets.

Pros:

  • Very efficient for problems where n is large and k is close to n.
  • No sorting or heap maintenance required, and it handles frequencies directly.

Cons:

  • Bucket sort can be less intuitive to implement.
  • Requires extra space for buckets, which may not be ideal for space-constrained environments.

Certainly! Here’s the Quickselect implementation for solving the Top K Frequent Elements problem.

4. Quickselect Solution

Explanation:

  • We start by building a frequency dictionary to count the occurrences of each element.
  • Then, we convert this dictionary into a list of tuples (element, frequency).
  • Using Quickselect, we partition the list of tuples so that the k most frequent elements are positioned in the first k spots in the list.
  • After partitioning, we return the elements from the first k positions.

Implementation:

import random
from collections import Counter
from typing import List, Tuple

def top_k_frequent_quickselect(nums: List[int], k: int) -> List[int]:
    # Step 1: Count frequencies
    freq_count = Counter(nums)
    # Convert the dictionary to a list of (element, frequency) pairs
    freq_items = list(freq_count.items())

    def partition(left: int, right: int, pivot_index: int) -> int:
        pivot_frequency = freq_items[pivot_index][1]
        # Move pivot to the end
        freq_items[pivot_index], freq_items[right] = freq_items[right], freq_items[pivot_index]
        store_index = left
        # Move all elements with frequency greater than pivot to the left
        for i in range(left, right):
            if freq_items[i][1] > pivot_frequency:
                freq_items[store_index], freq_items[i] = freq_items[i], freq_items[store_index]
                store_index += 1
        # Move pivot to its final place
        freq_items[right], freq_items[store_index] = freq_items[store_index], freq_items[right]
        return store_index

    def quickselect(left: int, right: int, k_smallest: int):
        if left == right:  # If the list contains only one element
            return
        # Select a random pivot index
        pivot_index = random.randint(left, right)
        # Partition the array around the pivot
        pivot_index = partition(left, right, pivot_index)
        # Recursively apply quickselect on the relevant half
        if k_smallest == pivot_index:
            return
        elif k_smallest < pivot_index:
            quickselect(left, pivot_index - 1, k_smallest)
        else:
            quickselect(pivot_index + 1, right, k_smallest)

    # Perform Quickselect for the k most frequent elements
    n = len(freq_items)
    quickselect(0, n - 1, k - 1)

    # Return the first k elements' values from freq_items
    return [item[0] for item in freq_items[:k]]

# Example usage
nums = [1, 1, 1, 2, 2, 3]
k = 2
print(top_k_frequent_quickselect(nums, k))  # Output: [1, 2]

Explanation of Key Steps:

  1. Partition Function:
    • Selects a pivot and rearranges elements such that all elements with frequency higher than the pivot are on the left, and all elements with frequency lower than the pivot are on the right.
    • This allows us to position elements based on their frequency.
  2. Quickselect Function:
    • Partitions the list around a pivot index until the k-th most frequent element is in the correct position.
    • This process allows us to get the top k frequent elements in average O(n) time without fully sorting the list.

Pros and Cons:

  • Pros: Efficient with an average time complexity of O(n), ideal for large lists.
  • Cons: The worst-case time complexity is O(n^2), though random pivot selection mitigates this in practice.

Summary

  • Sorting Solution: Simple but inefficient for large n, with O(n log n) complexity.
  • Heap Solution: Ideal for large n with small k, with O(n log k) complexity.
  • Bucket Sort Solution: Optimal for large n and frequency-based grouping, with O(n) complexity, but uses more space.

'ML Engineering > python' 카테고리의 다른 글

[Sort] Merge Sort  (0) 2025.01.11
Heap/Quickselect | K Closest Points to Origin  (0) 2024.10.26
Heap/Quickselect| Finds the k-th smallest/largest element(s) in the list  (0) 2024.10.26
08. Math  (0) 2024.08.07
07. Subset Techniques  (0) 2024.08.06

Here's a structured script for explaining how to solve the K Closest Points to Origin problem with various approaches, including brute-force, heap, and optimized techniques, along with pros and cons:


You:
"To solve the problem of finding the k closest points to the origin in a list of points, we can explore multiple approaches depending on the input size and desired efficiency. The goal is to calculate the Euclidean distance for each point from the origin (0, 0) and return the k points with the smallest distances.

  1. Brute-Force Approach (Sorting)
    The simplest approach is to calculate the distance of each point from the origin, store these distances in a list, and then sort the list based on these distances. We can then select the first k points from this sorted list.
    • Calculate the Euclidean distance squared for each point to avoid using sqrt (not necessary for comparing distances).
    • Sort the list of points based on the calculated distances.
    • Return the first k points from the sorted list.
    Pros:
    • Straightforward and easy to implement.
    Cons:
    • Sorting the list takes O(n log n) time, which can be inefficient for large n if only k smallest distances are required.
    • This approach doesn’t leverage the fact that we only need the closest k points, not the full sorted list.
  2. Steps:
  3. Heap Approach (Optimized for Efficiency)
    To improve efficiency, we can use a max-heap to store the k closest points while traversing the points list:
    • We first add the first k points to the max-heap based on their distances to the origin.
    • For each additional point, calculate its distance and compare it with the root of the max-heap (the largest distance among the k closest points so far). If the new point’s distance is smaller, we remove the root and insert the new point into the heap.
    • This ensures that we only keep k points in the heap at any time.
    Pros:
    • O(n log k) time complexity, which is more efficient than sorting when k is much smaller than n.
    • The heap helps us efficiently maintain the k closest points without sorting the entire list.
    Cons:
    • Implementing and managing a max-heap might add complexity, but Python’s heapq library can simplify this.
  4. Quickselect Algorithm (Alternative Optimal Solution)
    Another approach is to use the Quickselect algorithm, which is similar to quicksort but only partially sorts the array to find the k smallest elements:
    • Select a pivot point and partition the array around this pivot based on distance.
    • After partitioning, if the pivot position is k, then the first k points in the list are the closest.
    • Otherwise, recursively apply quickselect on the relevant half of the array until the first k points are found.
    Pros:
    • O(n) average time complexity, which is faster than heap-based solutions for large inputs.
    • No additional memory overhead since quickselect works in-place.
    Cons:
    • The worst-case time complexity is O(n^2), though choosing a random pivot or median-of-medians strategy can mitigate this.
    • It’s more complex to implement than the heap approach.
  5. Alternative Approach (Using Sorted Containers for Smaller Inputs)
    For scenarios where the input list of points is relatively small, using Python’s built-in sorted function or even a data structure like a SortedList (from sortedcontainers library) can simplify implementation.
    • Sort the points based on the distance in O(n log n) time.
    • Return the first k points.
    This is typically less efficient than heap-based solutions for large n but can be simpler for small datasets.

To summarize:

  • For a basic solution, sorting works fine but is not efficient for large n.
  • For larger inputs, the max-heap approach is optimal for maintaining the closest k points in O(n log k) time.
  • For very large inputs, Quickselect provides an efficient in-place solution with average O(n) time complexity.

Given these options, I would choose the max-heap approach for balancing efficiency and ease of implementation, or Quickselect for the absolute best average time complexity."


Here are Python implementations for each solution to the K Closest Points to Origin problem: sorting, heap, and quickselect.

1. Sorting Solution

This approach calculates the distances and sorts the points by distance.

from typing import List

def k_closest_sort(points: List[List[int]], k: int) -> List[List[int]]:
    # Sort points by distance (no need for sqrt as we only need relative distance)
    points.sort(key=lambda point: point[0]**2 + point[1]**2)
    # Return the first k points
    return points[:k]

# Example usage
points = [[1, 3], [-2, 2], [5, 8], [0, 1]]
k = 2
print(k_closest_sort(points, k))  # Output: k closest points

2. Heap Solution

This approach uses a max-heap to keep track of the k closest points.

import heapq
from typing import List

def k_closest_heap(points: List[List[int]], k: int) -> List[List[int]]:
    # Initialize a max-heap with the first k points (negative distance for max-heap behavior)
    max_heap = [(-point[0]**2 - point[1]**2, point) for point in points[:k]]
    heapq.heapify(max_heap)

    # Iterate over the remaining points
    for point in points[k:]:
        distance = -point[0]**2 - point[1]**2  # Negative for max-heap
        # If current point is closer than the farthest in the heap
        if distance > max_heap[0][0]:
            heapq.heappop(max_heap)
            heapq.heappush(max_heap, (distance, point))

    # Extract the points from the heap
    return [point for (_, point) in max_heap]

# Example usage
points = [[1, 3], [-2, 2], [5, 8], [0, 1]]
k = 2
print(k_closest_heap(points, k))  # Output: k closest points

3. Quickselect Solution

This approach uses Quickselect to partition the points so that the k closest points are in the first k positions of the list.

import random
from typing import List

def k_closest_quickselect(points: List[List[int]], k: int) -> List[List[int]]:
    def distance(point):
        return point[0]**2 + point[1]**2

    def partition(left, right, pivot_index):
        pivot_distance = distance(points[pivot_index])
        # Move pivot to the end
        points[pivot_index], points[right] = points[right], points[pivot_index]
        store_index = left
        # Move all points closer than pivot to the left
        for i in range(left, right):
            if distance(points[i]) < pivot_distance:
                points[store_index], points[i] = points[i], points[store_index]
                store_index += 1
        # Move pivot to its final place
        points[right], points[store_index] = points[store_index], points[right]
        return store_index

    def quickselect(left, right, k_smallest):
        if left == right:  # If the list contains only one element
            return
        # Select a random pivot index
        pivot_index = random.randint(left, right)
        # Partition the array around the pivot
        pivot_index = partition(left, right, pivot_index)
        # Recursively apply the quickselect
        if k_smallest == pivot_index:
            return
        elif k_smallest < pivot_index:
            quickselect(left, pivot_index - 1, k_smallest)
        else:
            quickselect(pivot_index + 1, right, k_smallest)

    # Perform quickselect for the k closest points
    quickselect(0, len(points) - 1, k)
    # Return the first k elements
    return points[:k]

# Example usage
points = [[1, 3], [-2, 2], [5, 8], [0, 1]]
k = 2
print(k_closest_quickselect(points, k))  # Output: k closest points

Summary:

  • Sorting Solution: Simple but not optimal for large lists (O(n log n)).
  • Heap Solution: Efficient for large lists when k is small (O(n log k)).
  • Quickselect Solution: Most efficient average time complexity for large lists (O(n) on average).

Certainly! Here’s an explanation for each solution to the K Closest Points to Origin problem:


1. Sorting Solution

Explanation:

  • In this approach, we start by calculating the Euclidean distance squared for each point from the origin. We square the distances (point[0]**2 + point[1]**2) instead of taking the square root to avoid unnecessary computations and because the relative order of distances remains unchanged without square roots.
  • We sort the list of points based on these squared distances in ascending order.
  • Finally, we return the first k points from the sorted list as they represent the closest points.

Time Complexity: O(n log n)

  • Sorting the list of n points is the main operation and takes O(n log n) time.

Space Complexity: O(1) (assuming in-place sorting)

Pros:

  • Simple to implement and straightforward to understand.
  • Works well for relatively small input sizes.

Cons:

  • Sorting the entire list of points is inefficient for large inputs when we only need k closest points.
  • Doesn’t take advantage of the fact that we only need a partial result (k points) and could be optimized further for larger inputs.

2. Heap Solution (Optimal for Larger Inputs)

Explanation:

  • This approach uses a max-heap to store the k closest points. A max-heap allows us to efficiently maintain the k smallest distances at any given time.
  • We start by adding the first k points from points into the max-heap. We store each point in the heap with its negative squared distance, as Python’s heapq module only supports a min-heap, and we need a max-heap behavior to keep the farthest element at the root.
  • For each remaining point, we calculate its squared distance. If this new distance is smaller than the largest distance in the max-heap (i.e., the current farthest point), we replace the root with the new point, keeping only the k closest points in the heap.
  • After processing all points, the heap contains exactly the k closest points.

Time Complexity: O(n log k)

  • Constructing and maintaining a max-heap with only k elements takes O(log k) time per insertion.
  • For n points, this results in O(n log k) time, which is much more efficient than sorting for large n when k is small.

Space Complexity: O(k)

  • The heap stores only k points at any time, which is efficient in terms of space.

Pros:

  • Efficient for large inputs where k is much smaller than n, as we don’t need to sort all points.
  • Uses constant space for the heap, making it memory-efficient.

Cons:

  • Slightly more complex to implement, particularly due to the max-heap setup using Python’s min-heap functionality.

3. Quickselect Solution (Optimal Average-Case Time Complexity)

Explanation:

  • This approach uses the Quickselect algorithm, which is similar to quicksort but focuses only on partitioning the list until we have the k smallest elements at the start of the list.
  • We first select a pivot point randomly and partition the points based on their distance from the origin. All points with distances less than the pivot’s distance are moved to the left, and all with greater distances to the right.
  • After partitioning, if the pivot position is k, the first k elements are the closest points, and we return them.
  • If the pivot position is less than k, we apply quickselect recursively to the right side, as we need more points to reach k. If the pivot position is greater than k, we apply quickselect on the left side.
  • This process continues until we find the first k closest points.

Time Complexity: O(n) on average

  • Each partition step reduces the number of elements we’re working with, giving an average-case time complexity of O(n).
  • However, in the worst case (e.g., when the pivot is always poorly chosen), this algorithm can degrade to O(n^2).

Space Complexity: O(1) for in-place operations, though the recursive calls can add up.

Pros:

  • The most efficient average-case solution for large inputs, particularly when k is close to n.
  • In-place operations mean no additional space is needed beyond the input list.

Cons:

  • The worst-case time complexity is O(n^2), though using a random pivot helps avoid this in practice.
  • More complex to implement than heap or sorting solutions.

Summary of Approach Suitability

  • Sorting: Best for smaller lists or if you need a simple solution without concern for optimal efficiency.
  • Heap: Ideal for large lists where k is small compared to n, balancing ease of implementation with optimal performance.
  • Quickselect: The most efficient solution for very large lists when you want the lowest average time complexity, but it requires careful handling to avoid worst-case scenarios.

Each approach provides a useful solution depending on the input size and constraints, showing flexibility in problem-solving.

'ML Engineering > python' 카테고리의 다른 글

[Sort] Merge Sort  (0) 2025.01.11
Heap/Quickselect | Top K Frequent Elements  (1) 2024.10.26
Heap/Quickselect| Finds the k-th smallest/largest element(s) in the list  (0) 2024.10.26
08. Math  (0) 2024.08.07
07. Subset Techniques  (0) 2024.08.06

https://leetcode.com/problems/kth-largest-element-in-an-array/?envType=company&envId=facebook&favoriteSlug=facebook-thirty-days

 

"To approach this problem, I like to first consider the brute-force method because it helps clarify how the solution works and when it might become inefficient based on the input size.

For instance, let's say the problem is to find the 'k' largest elements in an array. If the input size 'n' is relatively small—let's say around 100 elements or fewer—I might start by sorting the array. Sorting takes O(n log n) time, so with 100 elements, that would be approximately 700 operations, which is manageable. Once sorted, I can simply take the last 'k' elements, which gives the correct answer with minimal complexity.

However, if the input size grows significantly—say, 1 million elements—sorting the entire array now takes about 20 million operations (O(n log n) with n = 1,000,000). At this scale, sorting is inefficient, especially if we only need a few elements, like the top 10. In such cases, it’s better to optimize the solution.

That's where heaps come into play. I would use a min-heap to store only 'k' elements. As I traverse the array, I compare each element to the smallest element in the heap (in O(1) time) and, if it's larger, I replace it (in O(log k) time). By the end of the traversal, the heap contains the 'k' largest elements. The time complexity for this approach is O(n log k), which, for large inputs, drastically reduces the number of operations. For example, with 1 million elements and 'k' equal to 10, this approach would only take about 140,000 operations, which is much more efficient than 20 million.

In summary:

  1. If the input size is small, I start with a brute-force approach because sorting is simple and sufficient for manageable input sizes.
  2. But if the input size is large, like in the range of hundreds of thousands or millions, I switch to a heap-based approach, which efficiently handles the problem in logarithmic time relative to 'k'."

You:
"After initially considering a brute-force solution, depending on the input size, I would next evaluate the more optimal approaches. The heap approach is one of the most commonly used because it allows us to efficiently maintain the 'k' largest or smallest elements.

For example, if we’re looking for the 'k' largest elements, I would use a min-heap of size 'k'. As I traverse the array, I compare each element to the root of the heap:

  • If the element is larger than the heap’s root (the smallest element in the heap), I replace the root with the new element and re-heapify. This takes O(log k) time for each operation, resulting in a total time complexity of O(n log k) for 'n' elements. This is efficient when 'k' is small compared to 'n', and avoids the inefficiencies of sorting the entire array.

However, there are also other approaches depending on the problem constraints:

  1. Quickselect Algorithm
    If I need the exact 'k-th' largest element, and I don’t necessarily need the elements in sorted order, I might consider the quickselect algorithm. Quickselect is a partition-based algorithm, similar to quicksort, but it only partially sorts the array around the 'k-th' element. This method works in O(n) on average and is efficient for finding one element (or a range) without fully sorting the array. It’s particularly useful when we’re concerned only with the 'k-th' element, rather than all the 'k' largest.
  2. Bucket Sort
    For specific scenarios where the input range is limited (e.g., when elements are integers within a known small range), I could use bucket sort. This approach involves dividing the array into buckets and distributing the elements into these buckets based on their values. After filling the buckets, we can gather the top 'k' elements by traversing the highest buckets. This method can achieve O(n) time if the range of values is small relative to 'n'. It’s a great option when the problem constraints allow, especially if I know the data distribution is skewed or limited.

To summarize:

  • I start with a brute-force approach for small input sizes or as a baseline.
  • I then move to a heap-based solution for larger input sizes where I need efficient handling of the 'k' largest or smallest elements.
  • If I’m only interested in the 'k-th' element, I might use quickselect for a more direct approach with average O(n) complexity.
  • In specific cases, like when the value range is known and limited, bucket sort might be the best option, offering linear time complexity for certain distributions."

You:
"Quickselect is an efficient algorithm for finding the 'k-th' smallest or largest element in an unsorted list. It's based on the same idea as quicksort, but instead of sorting the entire list, we focus only on the part of the list that contains the element we're looking for. This reduces the average time complexity to O(n), making it much faster than sorting, which takes O(n log n).

Let me walk through the approach step-by-step:

  1. Initial Setup
    We begin by picking a pivot element from the list. This can be any element, but in practice, choosing the middle element or a random element often works well to avoid worst-case scenarios.
  2. Partitioning
    Similar to quicksort, we partition the array around the pivot:
    • Elements smaller than the pivot are placed on the left.
    • Elements greater than the pivot are placed on the right.
    The key here is that, after partitioning, the pivot is in its correct position relative to the rest of the array—everything on the left is smaller, and everything on the right is larger.
  3. Decide Which Half to Search
    After partitioning, we check the position of the pivot:
    • If the pivot is in the 'k-th' position (for 'k-th' smallest), we have found our element, and we can return it.
    • If the pivot’s position is greater than 'k', we know the 'k-th' smallest element lies in the left subarray, so we recursively apply the same logic to that subarray.
    • If the pivot’s position is less than 'k', we search the right subarray for the 'k-th' smallest element.
    This recursive process continues until we find the desired element.
  4. Time Complexity
    In the average case, each partition step reduces the problem size by half, resulting in a time complexity of O(n). However, in the worst case (if the pivot is poorly chosen), the algorithm can degrade to O(n^2), similar to quicksort. But with a good pivot selection strategy (like choosing a random pivot), we can generally avoid this scenario.

Example:
Let’s say we’re looking for the 3rd smallest element in the array [7, 2, 1, 6, 8, 5, 3, 4]. Using quickselect:

  1. Pick a pivot—let's choose 4.
  2. Partition the array: [2, 1, 3] | 4 | [7, 6, 8, 5]. Now 4 is in its correct position (index 3).
  3. Since we’re looking for the 3rd smallest element, which is still in the left partition [2, 1, 3], we recursively apply quickselect to that subarray.
  4. Eventually, we find that 3 is the 3rd smallest element.

To summarize:

  • Quickselect is highly efficient for finding the 'k-th' smallest or largest element with an average time complexity of O(n).
  • Instead of sorting the entire array, it partitions the array recursively, focusing only on the necessary part.
  • It's a great alternative when we only need a specific element, rather than sorting everything."

import random

def quickselect(arr, k):
    """
    Finds the k-th smallest element in the list `arr`.

    Parameters:
    - arr (list of int): The list of integers.
    - k (int): The index (1-based) of the element to find in sorted order.

    Returns:
    - int: The k-th smallest element.
    """
    # Convert k to 0-based index for simplicity
    k = k - 1

    def partition(left, right, pivot_index):
        pivot_value = arr[pivot_index]
        # Move pivot to the end
        arr[pivot_index], arr[right] = arr[right], arr[pivot_index]
        store_index = left
        # Move all smaller elements to the left
        for i in range(left, right):
            if arr[i] < pivot_value:
                arr[i], arr[store_index] = arr[store_index], arr[i]
                store_index += 1
        # Move pivot to its final place
        arr[right], arr[store_index] = arr[store_index], arr[right]
        return store_index

    def select(left, right):
        # Base case: the list contains only one element
        if left == right:
            return arr[left]

        # Select a random pivot index
        pivot_index = random.randint(left, right)

        # Perform partitioning
        pivot_index = partition(left, right, pivot_index)

        # The pivot is in its final sorted position
        if pivot_index == k:
            return arr[pivot_index]
        elif pivot_index < k:
            # Go right
            return select(pivot_index + 1, right)
        else:
            # Go left
            return select(left, pivot_index - 1)

    return select(0, len(arr) - 1)

Example Usage:

arr = [7, 2, 1, 6, 8, 5, 3, 4]
k = 3
print(f"The {k}-th smallest element is {quickselect(arr, k)}.")

Explanation:

  • partition: This function arranges elements around a pivot so that all smaller elements are to the left and all larger elements are to the right.
  • select: This recursive function narrows down the search based on the position of the pivot.
  • Time Complexity: On average, Quickselect takes O(n) time due to halving the search space with each partition.

'ML Engineering > python' 카테고리의 다른 글

Heap/Quickselect | Top K Frequent Elements  (1) 2024.10.26
Heap/Quickselect | K Closest Points to Origin  (0) 2024.10.26
08. Math  (0) 2024.08.07
07. Subset Techniques  (0) 2024.08.06
06. Top K Elements Technique  (0) 2024.08.06

In PyTorch, when defining a nn.Module, attributes can either be trainable parameters (like weights and biases) or non-trainable values (such as constants, buffers, or pre-computed values).

Trainable Parameters vs Non-Trainable Attributes:

  1. Trainable Parameters:
    • These are parameters that the model updates during training via backpropagation (e.g., weights in a neural network).
    • PyTorch stores these parameters using nn.Parameter and registers them to the model.
    • Example: weights and biases in layers like nn.Linear or nn.Conv2d.
    self.weight = nn.Parameter(torch.randn(10, 10))  # Trainable
  2. Non-Trainable Attributes:
    • These are attributes that do not change during training. They are useful for storing constants, lookup tables, pre-initialized matrices, etc.
    • If you don’t want these values to be updated via backpropagation, you typically register them as a buffer or store them as regular attributes of the module.
    • Example: a normalization constant, a precomputed matrix, or a codebook in vector quantization.
    self.constant = torch.randn(10, 10)  # Non-trainable, regular attribute

register_buffer:

  • PyTorch provides register_buffer to store non-trainable tensors in a model. This is useful because buffers will automatically be moved to the correct device (e.g., GPU) when the model is moved, but they won’t be updated during training.
  • However, if you don’t want or need this specific behavior, you can just store non-trainable values as regular attributes.
        def __init__(block_size):
            self.register_buffer("mask", torch.tril(torch.ones(block_size, block_size))
                    .view(1, 1, block_size, block_size)) 
        def forward(x): 
            B, T, C = x.size() 
            self.mask[:,:,:T,:T]

WebDataset and Lhotse are both tools designed to facilitate working with large-scale datasets, particularly in the context of machine learning with PyTorch.

In summary,

  • WebDataset for general-purpose, scalable data handling across multiple modalities
  • Lhotse for specialized speech and audio processing tasks where detailed data preparation is critical.

WebDataset

Overview:

  • Purpose: Primarily designed for streaming large datasets stored in tar archives in distributed training environments.
  • Data Format: Works with tar files containing various types of data, such as images, audio, and text.
  • Integration: Integrates directly with PyTorch’s DataLoader, making it easy to use in deep learning pipelines.
  • Key Features:
    • Streaming: Enables on-the-fly data loading from tar archives, reducing memory overhead.
    • Sharding: Supports data sharding across multiple GPUs or nodes, optimizing for distributed training.
    • Flexibility: Can handle multiple data types (images, audio, etc.) in a single archive.
    • Compression: Supports compression, which can save storage space and bandwidth during data loading.

Best For:

  • Large-scale, distributed training where data needs to be streamed from disk or cloud storage.
  • Projects requiring efficient handling of large datasets stored in tar archives.
  • Use cases where different types of data (e.g., images, audio, text) are stored together.

Lhotse

Overview:

  • Purpose: A toolkit specifically designed for preparing and managing large-scale speech and audio datasets, particularly for speech processing tasks.
  • Data Format: Works with various audio formats and annotations, supporting efficient data storage and access.
  • Integration: Also integrates with PyTorch, providing ready-to-use Dataset classes for speech recognition, speaker verification, and other audio tasks.
  • Key Features:
    • Data Preparation: Provides tools for preparing and managing datasets, including feature extraction, data augmentation, and metadata handling.
    • Rich Metadata Handling: Lhotse is highly optimized for working with audio datasets that include rich metadata, such as transcriptions, speaker labels, and more.
    • Feature Extraction: Includes utilities for extracting features like MFCCs, spectrograms, and more, commonly used in speech processing tasks.
    • Interoperability: Can work with existing datasets and tools, making it easy to integrate into existing workflows.

Best For:

  • Speech processing tasks, such as speech recognition, speaker verification, or speech synthesis.
  • Projects that require detailed handling of audio data and associated metadata.
  • Use cases where preprocessing (e.g., feature extraction) and dataset preparation are crucial components of the workflow.

Comparison Summary:

  • Focus:
    • WebDataset is more general-purpose, suitable for handling a variety of data types (e.g., images, audio, text) in large-scale, distributed training environments.
    • Lhotse is specialized for speech and audio processing, with extensive support for audio-specific data preparation, feature extraction, and metadata management.
  • Use Cases:
    • Use WebDataset if your project involves diverse types of large-scale data that need to be streamed efficiently during training, particularly in distributed setups.
    • Use Lhotse if your focus is on speech processing tasks, and you need robust tools for managing and preparing large audio datasets with detailed annotations.
  • Integration:
    • Both integrate well with PyTorch, but WebDataset focuses on data loading efficiency and scalability, while Lhotse provides a comprehensive toolkit for the entire data preparation process in speech tasks.

Lhotse is a Python toolkit designed to facilitate the preparation, processing, and management of large-scale speech and audio datasets, particularly for tasks in speech processing. Its comprehensive features for dataset preparation, feature extraction, and metadata management make it an invaluable tool for anyone working with large-scale speech and audio data. Whether you're developing ASR systems, speaker verification models, or other speech-related technologies, Lhotse provides the necessary tools to streamline and enhance your data processing workflows. It is named after Lhotse, the fourth highest mountain in the world, reflecting its goal to handle large and complex audio data efficiently.

 

Key Features:

  • Dataset Preparation:
    • Lhotse provides a comprehensive set of tools for preparing speech datasets, including downloading, organizing, and processing audio data.
    • It supports various audio formats (e.g., WAV, MP3, FLAC) and can handle different sampling rates and channel configurations.
  • Feature Extraction:
    • The toolkit includes utilities for extracting common audio features used in speech processing, such as Mel-frequency cepstral coefficients (MFCCs), filter banks, and spectrograms.
    • These features are crucial for tasks like ASR and are compatible with machine learning models.
  • Rich Metadata Handling:
    • Lhotse allows for the detailed management of metadata associated with audio files, such as transcriptions, speaker labels, and timing information (e.g., start and end times of utterances).
    • This capability is particularly important for tasks requiring alignment between audio and text, such as speech recognition.
  • Data Augmentation:
    • The toolkit includes built-in support for data augmentation techniques, such as speed perturbation and noise injection, which are commonly used to improve the robustness of speech models.
  • Interoperability:
    • Lhotse is designed to be compatible with existing datasets and tools in the speech processing ecosystem. It can work with popular datasets like LibriSpeech, VoxCeleb, and others.
    • It also integrates smoothly with PyTorch, providing ready-to-use Dataset classes that can be directly employed in training pipelines.
  • Scalability and Efficiency:
    • Lhotse is optimized for efficiency, handling large datasets and extensive metadata without becoming a bottleneck in the data processing pipeline.
    • It supports lazy loading and caching, which helps in managing memory usage and speeding up data access during training.

WebDataset is a PyTorch-compatible library designed to streamline the process of working with large-scale datasets stored in archive formats, such as tar files. It is particularly useful for training deep learning models in distributed environments, where efficient data loading and processing are critical.

 

Key Features:

  • Streaming and Sharding: WebDataset allows you to stream data directly from tar archives, making it ideal for large datasets that don't fit into memory. It also supports sharding, which helps in distributing the data across multiple GPUs or nodes, facilitating parallel processing.
  • Flexible Data Formats: You can store various types of data (e.g., images, audio, text) within the same tar archive, and the library can handle these different formats seamlessly. This flexibility makes it suitable for complex machine learning tasks that involve multi-modal data.
  • Integration with PyTorch DataLoader: WebDataset integrates smoothly with PyTorch's DataLoader, enabling efficient and scalable data pipelines. You can easily create custom datasets that load and preprocess data on-the-fly during training.
  • Performance Optimization: By leveraging streaming, compression, and parallel processing, WebDataset helps minimize I/O bottlenecks and maximizes training throughput, which is especially beneficial in large-scale, distributed training scenarios.
  •  

Use Cases:

  • Distributed Training: WebDataset is often used in scenarios where training needs to be distributed across multiple GPUs or machines, making it easier to manage large datasets efficiently.
  • Large-Scale Image or Audio Processing: It’s particularly useful for projects that involve massive collections of images or audio files, where data needs to be processed quickly and efficiently.
  • Data Pipelines in the Cloud: The streaming capability of WebDataset also makes it suitable for cloud-based environments, where data can be streamed directly from cloud storage services without needing to download everything first.

 

+ Recent posts