Приведи примеры алгоритмов сортировки с их асимптотической сложностью

Ответ

Вот несколько классических алгоритмов сортировки с их временной сложностью:

1. Сортировка пузырьком (Bubble Sort)

  • Сложность: O(n²) в худшем и среднем случае, O(n) в лучшем (уже отсортированный массив)
  • Применение: Учебный пример, не используется на практике из-за низкой эффективности
def bubble_sort(arr):
    n = len(arr)
    for i in range(n):
        swapped = False
        for j in range(0, n-i-1):
            if arr[j] > arr[j+1]:
                arr[j], arr[j+1] = arr[j+1], arr[j]
                swapped = True
        if not swapped:
            break
    return arr

2. Быстрая сортировка (Quick Sort)

  • Сложность: O(n log n) в среднем, O(n²) в худшем (при неудачном выборе опорного элемента)
  • Применение: Один из самых быстрых алгоритмов на практике, используется в стандартных библиотеках
def quick_sort(arr):
    if len(arr) <= 1:
        return arr
    pivot = arr[len(arr) // 2]
    left = [x for x in arr if x < pivot]
    middle = [x for x in arr if x == pivot]
    right = [x for x in arr if x > pivot]
    return quick_sort(left) + middle + quick_sort(right)

3. Сортировка слиянием (Merge Sort)

  • Сложность: O(n log n) во всех случаях
  • Применение: Гарантированная сложность, стабильная сортировка, хороша для связных списков
def merge_sort(arr):
    if len(arr) <= 1:
        return arr

    mid = len(arr) // 2
    left = merge_sort(arr[:mid])
    right = merge_sort(arr[mid:])

    return merge(left, right)

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

    while i < len(left) and j < len(right):
        if left[i] < right[j]:
            result.append(left[i])
            i += 1
        else:
            result.append(right[j])
            j += 1

    result.extend(left[i:])
    result.extend(right[j:])
    return result

4. Timsort (используется в Python)

  • Сложность: O(n log n) в худшем случае, O(n) в лучшем
  • Применение: Гибридный алгоритм, сочетающий сортировку вставками и слиянием. Это стандартный алгоритм в Python (list.sort() и sorted())

Выбор алгоритма зависит от:

  • Размера данных
  • Требований к стабильности
  • Распределения данных
  • Доступной памяти

Ответ 18+ 🔞

А, ну вот, классику вспомнил! Алгоритмы сортировки, ёпта. Сидишь такой, пишешь код, а тут бац — и нужно всё это богатство по полочкам разложить. Давай разберём, что к чему, без этой академической зауми, на пальцах.

1. Сортировка пузырьком (Bubble Sort) Это просто пиздец, а не алгоритм, честно говоря. Представь: ты стоишь в очереди за водкой, а какой-то мудак постоянно сравнивает себя с соседом и, если он больше, меняется местами. И так до тех пор, пока самый большой мужик не окажется в конце. Сложность у него — O(n²), то есть если элементов овердохуища, то он будет работать до второго пришествия. В лучшем случае, если массив уже отсортирован, он просто пробежится и скажет «окей» — это O(n). Используют его только для мучения студентов, на практике — хуй с горы.

def bubble_sort(arr):
    n = len(arr)
    for i in range(n):
        swapped = False
        for j in range(0, n-i-1):
            if arr[j] > arr[j+1]:
                arr[j], arr[j+1] = arr[j+1], arr[j]
                swapped = True
        if not swapped:
            break
    return arr

2. Быстрая сортировка (Quick Sort) Вот это уже дело! Алгоритм с характером, ядраная вошь. Берёт какой-то элемент (опорный), всех, кто меньше, — кидает налево, кто больше — направо, а потом рекурсивно так же долбит эти куски. Сложность в среднем — O(n log n), и это охуенно быстро. Но есть подвох: если постоянно невезуче выбирать опорный элемент (например, самый маленький в уже отсортированном массиве), то доверия ебать ноль, и сложность скатится до O(n²). В реальности его обожают, он в куче библиотек сидит.

def quick_sort(arr):
    if len(arr) <= 1:
        return arr
    pivot = arr[len(arr) // 2]
    left = [x for x in arr if x < pivot]
    middle = [x for x in arr if x == pivot]
    right = [x for x in arr if x > pivot]
    return quick_sort(left) + middle + quick_sort(right)

3. Сортировка слиянием (Merge Sort) Стабильный, предсказуемый парень. Работает по принципу «разделяй и властвуй»: бьёт массив пополам, пока не останутся кусочки по одному элементу (это уже отсортировано, лол), а потом сливает их обратно в правильном порядке. Сложность всегда O(n log n), хоть тресни. За это его и любят, особенно для связных списков или когда нужна стабильность (чтобы равные элементы не прыгали). Но памяти он жрёт — мама не горюй.

def merge_sort(arr):
    if len(arr) <= 1:
        return arr

    mid = len(arr) // 2
    left = merge_sort(arr[:mid])
    right = merge_sort(arr[mid:])

    return merge(left, right)

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

    while i < len(left) and j < len(right):
        if left[i] < right[j]:
            result.append(left[i])
            i += 1
        else:
            result.append(right[j])
            j += 1

    result.extend(left[i:])
    result.extend(right[j:])
    return result

4. Timsort (тот самый, из Python) А это, бля, наш, родной! Гибридная хитрая жопа, которая сочетает сортировку вставками для мелких кусков и слияние для больших. Придумали для реальной жизни, где данные часто частично упорядочены. Сложность — O(n log n) в худшем случае, а если повезёт и данные почти отсортированы, то вообще O(n). Именно поэтому в Python list.sort() и sorted() — это огонь. Удивление пиздец, как умно придумано.

Так какой же алгоритм выбрать? Да э бошка думай, исходя из ситуации:

  • Размер данных: Для пяти элементов и пузырёк сойдёт, а для миллионов — быстрая или Timsort.
  • Нужна стабильность? Если порядок равных элементов важен — слияние или Timsort.
  • Как данные лежат? Если почти отсортированы — Timsort обнимет и приласкает.
  • Память есть? Быстрая сортировка на месте экономит память, а слияние — нет.

Короче, терпения ноль ебать учить эту теорию, но без неё потом в проектах будешь как полупидор с пузырьковой сортировкой на проде. Не будь таким.