[알고리즘]힙 정렬(Heap Sort)

2023. 7. 18. 11:40·CS 지식/[알고리즘]

완전 이진 트리란?

삽입할 때 왼쪽부터 차례대로 추가하는 이진 트리

힙 정렬

  • 완전이진 트리를 이용한 정렬 방식
  • 평균과 최악 모두 시간 복잡도는 O(nlog2n)

과정

  1. 최대 힙을 구성
  2. 현재 힙 루트는 가장 큰 값이 존재함. 루트의 값을 마지막 요소와 바꾼 후, 힙의 사이즈 하나 줄임
  3. 힙의 사이즈가 1보다 크면 위 과정 반복

루트를 마지막 노드로 대체 (11 → 4), 다시 최대 힙 구성

이와 같은 방식으로 최대 값을 하나씩 뽑아내면서 정렬하는 것이 힙 소트

public void heapSort(int[] array) {
    int n = array.length;
    
    // max heap 초기화
    for (int i = n/2-1; i>=0; i--){
        heapify(array, n, i); // 1
    }
    
    // extract 연산
    for (int i = n-1; i>0; i--) {
        swap(array, 0, i); 
        heapify(array, i, 0); // 2
    }
}

1번째 heapify

일반 배열을 힙으로 구성하는 역할

자식노드로부터 부모노드 비교

  • n/2-1부터 0까지 인덱스가 도는 이유는? 부모 노드의 인덱스를 기준으로 왼쪽 자식노드 (i2 + 1), 오른쪽 자식 노드(i2 + 2)이기 때문

2번째 heapify

요소가 하나 제거된 이후에 다시 최대 힙을 구성하기 위함

루트를 기준으로 진행(extract 연산 처리를 위해)

public void heapify(int array[], int n, int i) {
    int p = i;
    int l = i*2 + 1;
    int r = i*2 + 2;
    
    //왼쪽 자식노드
    if (l < n && array[p] < array[l]) {
        p = l;
    }
    //오른쪽 자식노드
    if (r < n && array[p] < array[r]) {
        p = r;
    }
    
    //부모노드 < 자식노드
    if(i != p) {
        swap(array, p, i);
        heapify(array, n, p);
    }
}

다시 최대 힙을 구성할 때까지 부모 노드와 자식 노드를 swap하며 재귀 진행

퀵 정렬과 합병 정렬의 성능이 좋기 때문에 힙 정렬의 사용 빈도가 높지는 않음.

하지만 힙 자료구조가 많이 활용되고 있으며, 이때 함께 따라오는 개념이 힙 소트

힙 소트가 유용할 때

  • 가장 크거나 가장 작은 값을 구할 때최소 힙 or 최대 힙의 루트 값이기 때문에 한번의 힙 구성을 통해 구하는 것이 가능
  • 최대 k 만큼 떨어진 요소들을 정렬할 때삽입 정렬보다 더욱 개선된 결과를 얻어낼 수 있음
  private void solve() {
      int[] array = { 230, 10, 60, 550, 40, 220, 20 };

      heapSort(array);

      for (int v : array) {
          System.out.println(v);
      }
  }

  public static void heapify(int array[], int n, int i) {
      int p = i;
      int l = i * 2 + 1;
      int r = i * 2 + 2;

      if (l < n && array[p] < array[l]) {
          p = l;
      }

      if (r < n && array[p] < array[r]) {
          p = r;
      }

      if (i != p) {
          swap(array, p, i);
          heapify(array, n, p);
      }
  }

  public static void heapSort(int[] array) {
      int n = array.length;

      // init, max heap
      for (int i = n / 2 - 1; i >= 0; i--) {
          heapify(array, n, i);
      }

      // for extract max element from heap
      for (int i = n - 1; i > 0; i--) {
          swap(array, 0, i);
          heapify(array, i, 0);
      }
  }

  public static void swap(int[] array, int a, int b) {
      int temp = array[a];
      array[a] = array[b];
      array[b] = temp;
  }
저작자표시 비영리 변경금지 (새창열림)

'CS 지식 > [알고리즘]' 카테고리의 다른 글

[알고리즘] 선택 정렬(Selection Sort)  (0) 2023.07.18
[알고리즘] 버블 정렬(Bubble Sort)  (1) 2023.07.18
[알고리즘]퀵 정렬(Quick Sort)  (0) 2023.07.18
[알고리즘] 삽입 정렬(Insertion Sort)  (1) 2023.07.18
[알고리즘] 빅오표기법(big-O notation)  (0) 2023.07.17
'CS 지식/[알고리즘]' 카테고리의 다른 글
  • [알고리즘] 버블 정렬(Bubble Sort)
  • [알고리즘]퀵 정렬(Quick Sort)
  • [알고리즘] 삽입 정렬(Insertion Sort)
  • [알고리즘] 빅오표기법(big-O notation)
woojin._.
woojin._.
여러가지 개발을 해보며 발생하는 이야기들에 대한 블로그입니다:)
  • woojin._.
    Jin's Dev Story
    woojin._.
  • 전체
    오늘
    어제
    • 분류 전체보기 (829)
      • Tools (25)
        • eGovFrame (3)
        • GeoServer (3)
        • QGIS (2)
        • LabelImg (2)
        • Git (6)
        • GitHub (1)
        • Eclipse (7)
        • Visual Studio (1)
      • Web & Android (121)
        • SpringBoot (37)
        • Three.js (2)
        • Spring Data JPA (9)
        • 스프링 부트 쇼핑몰 프로젝트 with JPA (25)
        • Thymeleaf (4)
        • Spring Security (15)
        • Flutter (29)
      • Programming Language (61)
        • JAVA (27)
        • JavaScript (14)
        • Dart (2)
        • Python (15)
        • PHP (3)
      • Database (43)
        • PostgreSQL (32)
        • MYSQL (7)
        • Oracle (3)
        • MSSQL (1)
      • SERVER (17)
        • TCP_IP (3)
        • 리눅스 (7)
        • AWS (7)
      • Coding Test (445)
        • 백준[JAVA] (108)
        • 프로그래머스[JAVA] (260)
        • 알고리즘 고득점 Kit[JAVA] (3)
        • SQL 고득점 Kit[ORACLE] (74)
      • CS 지식 (49)
        • [자료구조] (14)
        • [네트워크] (12)
        • [데이터베이스] (10)
        • [알고리즘] (9)
        • [운영체제] (4)
      • 기타 (6)
      • 자격증 & 공부 (62)
        • 정보처리기사 (2)
        • SQLD (6)
        • 네트워크관리사 2급 (5)
        • 리눅스마스터 1급 (44)
        • 리눅스마스터 2급 (1)
        • ISTQB (3)
        • 시스템보안 (1)
  • 블로그 메뉴

    • 홈
    • 태그
    • 방명록
  • 인기 글

  • 태그

    postgresql
    백준
    programmers
    리눅스마스터
    자바
    스프링 부트 쇼핑몰 프로젝트 with JPA
    Spring Security
    pcce 기출문제
    스프링부트
    spring
    시큐리티
    Java
    Oracle
    데이터
    Linux
    프로그래머스
    리눅스
    python
    CS지식
    CS
    Flutter
    springboot
    플러터
    JPA
    데이터베이스
    리눅스마스터 1급
    DB
    backjoon
    baekjoon
    스프링
  • 최근 글

  • hELLO· Designed By정상우.v4.10.0
woojin._.
[알고리즘]힙 정렬(Heap Sort)
상단으로

티스토리툴바