algorithm/common

정렬 Sort 알고리즘 구현

아르르르를를르 2020. 3. 5. 22:37

5가지 sort python으로 구현

SelectionSort, InsertionSort, BubbleSort, MergeSort, QuickSort

#-*- coding: utf-8 -*-

def selectionSort(arr):
    """
    시간 O(n^2)
    공간 O(n)
    (현재위치에 값을 찾음)
    정렬되지 않은 리스트를 첫번째 index에서부터 시작
    해당 index값을 포함하여 그 뒤 값들과 비교하고 최소값과 위치를 바꾼다. (swap)
    """
    for i in range(len(arr) - 1):
        min_idx = i
        for j in range(i + 1, len(arr)):
            if arr[j] < arr[min_idx]:
                min_idx = j
        arr[i], arr[min_idx] = arr[min_idx], arr[i]
    return arr

def insertionSort(arr):
    """
    시간 O(n^2) -> 정렬이 되어있는경우 O(n)
    공간 O(n)
    (현재위치에서 그 이전값들을 비교하여 맞는 자리에 삽입함)
    두번째 index부터 시작한다.
    """
    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
    return arr

def bubbleSort(arr):
    """
    시간 O(n^2)
    공간 O(n)
    (매번 연속된 index를 비교하여 큰 값을 위로 올림)
    두번째 index부터 시작한다.
    """
    for i in range(0, len(arr)):
        for j in range(1, len(arr)-i):
            if arr[j-1] > arr[j]:
                arr[j-1], arr[j] = arr[j], arr[j-1]
    return arr

def mergeSort(arr):
    """
    시간 O(NlogN)
    공간 O(2N)
    (Divide and conquer)
    Divide
    배열크기가 1일때까지 쪼갠다.
    conquer
    배열 2개 A, B를 앞에서부터 비교해서 새로운 배열 C에 넣는다.
    """
    if len(arr) <= 1:
        return arr
    mid = len(arr) // 2
    left = arr[:mid]
    right = arr[mid:]
    left = mergeSort(left)
    right = mergeSort(right)
    return merge(left, right)

def merge(left, right):
    total = []
    while(len(left)>0 or len(right)>0):
        try:
            if left[0] < right[0]:
                total.append(left[0])
                left = left[1:]
            else:
                total.append(right[0])
                right = right[1:]
        except IndexError:
            if len(left) == 0:
                total.append(right[0])
                right = right[1:]
            else:
                total.append(left[0])
                left = left[1:]
    return total
        

def quickSort(arr):
    """
    시간 O(NlogN) 분할깊이 logN, 이미 정렬된 경우 최악 O(N^2) 분할깊이 N
    공간 O(N)
    (Divide and conquer)
    랜덤으로 pivot 값을 기준으로 right, left
    pivot 기준으로 left < pivot, pivot < right
    left, right에서 각각 pivot 또 선택, pivot 기준으로 값 정렬
    list의 변동이 없으면 끝
    """
    if len(arr) <= 1: 
        return arr
    pivot = arr[len(arr) // 2] # 그러나 여기서 만약 pivot 이 첫번째라면?? - 상관없
    left, right, equal = [], [], []
    for item in arr:
        if item < pivot:
            left.append(item)
        elif item > pivot:
            right.append(item)
        else:
            equal.append(item)
    return quickSort(left) + equal + quickSort(right)

if __name__ == '__main__':
    sortedList = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]

    sampleList = [1, 10, 3, 5, 7, 4, 6, 2, 9, 8, 0]
    if selectionSort(sampleList) != sortedList:
        print("selectionSort Error")

    sampleList = [1, 10, 3, 5, 7, 4, 6, 2, 9, 8, 0]
    if insertionSort(sampleList) != sortedList:
        print("insertionSort Error")

    sampleList = [1, 10, 3, 5, 7, 4, 6, 2, 9, 8, 0]
    if bubbleSort(sampleList) != sortedList:
        print("bubbleSort Error")

    sampleList = [1, 10, 3, 5, 7, 4, 6, 2, 9, 8, 0]
    if mergeSort(sampleList) != sortedList:
        print("mergeSort Error")

    sampleList = [1, 10, 3, 5, 7, 4, 6, 2, 9, 8, 0]
    if quickSort(sampleList) != sortedList:
        print("quickSort Error")