View

기수 정렬  : 데이터가 모두 k 자릿수 이하의 자연수인 특수한 경우에 사용 가능

우선 가장 낮은 자릿수만 가지고 모든 수를 재배열(정렬)한다. 그런 다음 가장 낮은 자릿수는 잊어버리고 같은 방법으로 더 이상 자릿수가 남지 않을 때까지 계속한다.

시간 복잡도 : O(N)

 

 

계수 정렬  : 데이터의 크기 범위가 제한되어 정수 형태로 표현할 수 있을 때 사용 가능

⚠️  계수정렬은 직접 데이터의 값을 비교한 뒤에 위치를 변경하며 정렬하는 방식 (비교 기반의 정렬 알고리즘)이 아니다.

 

1) 가장 큰 데이터 + 1 크기의 모든 원소가 담길 수 있는 하나의 리스트를 생성

2)  처음에는 리스트의 모든 데이터가 0이 되도록 초기화한다.

3) 그 다음 데이터를 하나씩 확인하며 데이터의 값과 동일한 인덱스의 데이터를 1씩 증가

4) 입력 배열의 각 원소에 대해 count 리스트에 어느 인덱스에 들어가야하는지 조회

 

( 참고 애니메이션 : https://www.cs.miami.edu/home/burt/learning/Csc517.091/workbook/countingsort.html )

package sorting;

public class sorting04 {
    public static final int MAX_VALUE = 9;

    public static void main(String[] args) {
        int[] arr = {7, 5, 9, 0, 3, 1, 6, 2, 9, 1, 4, 8, 0, 5, 2};
        int[] cnt = new int[MAX_VALUE + 1];

        for (int i = 0; i < arr.length; i++) {
            cnt[arr[i]] += 1; // 각 데이터에 해당하는 인덱스의 값 증가
        }
        for (int i = 0; i <= MAX_VALUE; i++) {
            for (int j = 0; j < cnt[i]; j++) {
                System.out.print(i + " ");
            }
        }
    }
}

시간 복잡도 : O(N+K)

데이터의 개수 : N, 데이터 중 최대값의 크기 : K

리스트의 각 인덱스에 해당하는 값들을 확인할 때 데이터 중 최댓값의 크기만큼 반복을 수행해야 함

따라서 데이터의 범위만 한정되어 있다면 효과적으로 사용할 수 있으며 항상 빠르게 동작한다.

현존하는 정렬 알고리즘 중에서 기수 정렬과 더불어 가장 빠름

 

공간 복잡도 

계수 정렬은 때에 따라서 심각한 비효율성을 초래함. (ex. 데이터 수는 적으나 최소 최대값의 차이가 클 때)

동일한 값을 가지는 데이터가 여러 개 등장할 떄 적합 (ex. 100점 맞은 학생이 여러 명인 성적 정렬)

반면에 앞서 설명한 퀵 정렬은 일반적인 경우에서 평균적으로 빠르게 동작하기 때문에 데이터의 특성을 파악하기 어렵다면 퀵 정렬을 이용하는 것이 유리함

 

즉, 계수 정렬은

1. 데이터의 크기가 한정되어 있고

2. 데이터의 크기가 많이 중복되어 있을 때 유리

 

정렬 라이브러리

package sorting;

public class sorting04 {
    public static final int MAX_VALUE = 9;

    public static void main(String[] args) {
        int[] arr = {7, 5, 9, 0, 3, 1, 6, 2, 9, 1, 4, 8, 0, 5, 2};
        int[] cnt = new int[MAX_VALUE + 1];

        for (int i = 0; i < arr.length; i++) {
            cnt[arr[i]] += 1; // 각 데이터에 해당하는 인덱스의 값 증가
        }
        for (int i = 0; i <= MAX_VALUE; i++) {
            for (int j = 0; j < cnt[i]; j++) {
                System.out.print(i + " ");
            }
        }
    }
}

정렬 라이브러리는 항상 최악의 경우에도 O(NlogN)을 보장한다.

사실 우리가 직접 퀵 정렬을 구현할 때보다 더욱더 효과적임

(문제에서 별도의 요구가 없이 단순 정렬해야하는 경우  > 기본 정렬 라이브러리 사용

데이터의 범위가 한정되어 있으며 더 빠르게 동작해야 할 때 > 계수 정렬)

 

📚 실전문제 ) 위에서 아래로

수열을 내림차순으로 정렬하는 프로그램을 만드시오

package sorting;

import java.util.*;

public class sorting06 {
    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);

        int n = sc.nextInt();

        Integer[] arr = new Integer[n];
        for (int i = 0; i < n; i++) {
            arr[i] = sc.nextInt();
        }

        Arrays.sort(arr, Collections.reverseOrder());

        for(int i = 0; i < n; i++) {
            System.out.print(arr[i] + " ");
        }
    }
}

컬렉션이란? 자바에서 자료구조를 표현하는 인터페이스

Collection의 자료형에는 기본 타입은 올 수 없다.

더보기

자바의 자료형은 크게 기본 타입(primitive type)과 참조 타입(reference type)으로 나누어진다.

-기본 타입 : char, int, float, double, boolean 등

- 참조 타입 : String, Integer, Boolean, List 등 클래스 및 인터페이스

 

일반적으로 코딩 테스트에서 정렬 알고리즘이 사용되는 경우
1. 정렬 라이브러리로 풀 수 있는 문제 : 단순히 정렬 기법을 알고 있는지 물어보는 문제
2. 정렬 알고리즘의 원리에 대해서 물어보는 문제 : 선택/삽입/퀵 정렬 등의 원리를 알아야함
3. 더 빠른 정렬이 필요한 문제 : 퀵 정렬 기반의 정렬 기법으로는 풀 수 없으며 계수 정렬 등의 다른 정렬 알고리즘을 이용하거나 문제에서 기존에 알려진 알고리즘의 구조적인 개선을 거쳐야 풀 수 있다.
728x90

'알고리즘' 카테고리의 다른 글

[알고리즘] 해시맵(HashMap)  (0) 2023.02.07
[알고리즘] Scanner VS BufferedReader  (0) 2022.11.28
우선순위 큐(Prioriy Queue)  (0) 2022.11.17
Share Link
reply
«   2024/10   »
1 2 3 4 5
6 7 8 9 10 11 12
13 14 15 16 17 18 19
20 21 22 23 24 25 26
27 28 29 30 31