Ответ
Вот несколько классических алгоритмов сортировки с их временной сложностью:
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 обнимет и приласкает.
- Память есть? Быстрая сортировка на месте экономит память, а слияние — нет.
Короче, терпения ноль ебать учить эту теорию, но без неё потом в проектах будешь как полупидор с пузырьковой сортировкой на проде. Не будь таким.