개발이야기/파이썬
[Python - 알고리즘 ] 정렬
Study & Stack
2025. 5. 26. 17:25
728x90
정렬 알고리즘 완벽 정리
1. 정렬이란?
정렬(Sorting)은 데이터를 특정 기준(보통 오름차순, 내림차순)에 따라 재배열하는 작업입니다. 효율적인 검색, 탐색, 중복 제거 등 다양한 알고리즘과 시스템에서 필수적입니다.
2. 주요 정렬 알고리즘
1) 버블 정렬 (Bubble Sort)
인접한 두 원소를 비교해 크기가 순서대로 되어 있지 않으면 서로 교환하며 정렬합니다.
def bubble_sort(arr): n = len(arr) for i in range(n): for j in range(0, n - i - 1): # 인접 원소 비교 if arr[j] > arr[j + 1]: # 자리 교환 arr[j], arr[j + 1] = arr[j + 1], arr[j]
시간복잡도: 최악, 평균 O(n²), 공간복잡도: O(1)
2) 선택 정렬 (Selection Sort)
매 회 반복마다 아직 정렬되지 않은 부분에서 최소값을 찾아 맨 앞 원소와 교환합니다.
def selection_sort(arr): n = len(arr) for i in range(n): min_idx = i for j in range(i + 1, n): if arr[j] < arr[min_idx]: min_idx = j arr[i], arr[min_idx] = arr[min_idx], arr[i]
시간복잡도: 최악, 평균 O(n²), 공간복잡도: O(1)
3) 삽입 정렬 (Insertion Sort)
두 번째 원소부터 앞의 원소들과 비교하며 알맞은 위치에 삽입해 정렬합니다.
def insertion_sort(arr): for i in range(1, len(arr)): key = arr[i] j = i - 1 while j >= 0 and arr[j] > key: arr[j + 1] = arr[j] j -= 1 arr[j + 1] = key
시간복잡도: 최악, 평균 O(n²), 공간복잡도: O(1)
4) 병합 정렬 (Merge Sort)
분할 정복(divide and conquer) 알고리즘으로, 배열을 반으로 나누고 각각 정렬한 후 합칩니다.
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
시간복잡도: 최악, 평균 O(n log n), 공간복잡도: O(n)
5) 퀵 정렬 (Quick Sort)
피벗을 기준으로 작거나 같은 그룹과 큰 그룹으로 분할해 재귀적으로 정렬합니다.
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)
시간복잡도: 평균 O(n log n), 최악 O(n²), 공간복잡도: O(log n)~O(n)
3. 정렬 응용 문제
1) 두 배열 합치기 (병합 정렬 아이디어 활용)
두 개의 정렬된 배열을 하나의 정렬된 배열로 병합합니다.
def merge_two_sorted_lists(a, b): i = j = 0 merged = [] while i < len(a) and j < len(b): if a[i] < b[j]: merged.append(a[i]) i += 1 else: merged.append(b[j]) j += 1 # 남은 요소 추가 merged.extend(a[i:]) merged.extend(b[j:]) return merged # 테스트 print(merge_two_sorted_lists([1,3,5], [2,4,6])) # [1,2,3,4,5,6]
2) K번째 큰 수 찾기 (퀵 셀렉트 알고리즘)
정렬하지 않고 퀵 정렬 피벗 아이디어를 활용해 효율적으로 K번째 큰 수를 찾는 문제입니다.
def quick_select(arr, k): if len(arr) == 1: return arr[0] 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] if k <= len(left): return quick_select(left, k) elif k <= len(left) + len(middle): return pivot else: return quick_select(right, k - len(left) - len(middle)) # 테스트: 3번째 큰 수 찾기 print(quick_select([7,10,4,3,20,15], 3)) # 결과: 10
3) 회의실 배정 문제 (그리디 + 정렬)
회의가 끝나는 시간을 기준으로 정렬 후 최대한 많은 회의를 배정하는 문제
def max_meetings(intervals): intervals.sort(key=lambda x: x[1]) # 종료 시간 기준 정렬 count = 0 end_time = 0 for start, end in intervals: if start >= end_time: count += 1 end_time = end return count # 테스트 print(max_meetings([(1,4), (2,3), (3,5), (7,8)])) # 결과: 3
728x90