알고리즘

BOJ 1715 - 카드 정렬하기(Java)

장중앙 2021. 12. 27. 18:54

https://www.acmicpc.net/problem/1715

 

1715번: 카드 정렬하기

정렬된 두 묶음의 숫자 카드가 있다고 하자. 각 묶음의 카드의 수를 A, B라 하면 보통 두 묶음을 합쳐서 하나로 만드는 데에는 A+B 번의 비교를 해야 한다. 이를테면, 20장의 숫자 카드 묶음과 30장

www.acmicpc.net

 

풀이

이전의 합이 누적되는 형태

    - (a + b) + (a + b + c) + (a + b + c + d) + ......

    - 각 합의 작은 값이 누적되는 횟수를 크게, 큰 값이 누적되는 횟수를 작게 해야함

 

불가능한 방법

입력 요소의 정렬 만으로는 해결 불가능 

    - ex) 60, 50, 70, 80

        - 요소 정렬 : 50, 60, 70, 80

        - 누적 합 -> (50 + 60) + (50 + 60 + 70) + (50 + 60 + 70 + 80) = 560

    - 남아있는 수인 70, 80보다 큰 (50 + 60)인 110이 누적되는 것은 비효율적

 

해결방법

계산마다 요소들 중 가장 작은 값이 누적되게 해야함, 계산 할 때마다 정렬이 필요-> 우선순위 큐

    - 모든 요소 우선순위 큐에 삽입 -> 50, 60, 70, 80

    - 우선순위 큐에 사이즈가 1이 될때까지 앞의 2개 요소를 더하고 큐에 삽입

        - 비교 횟수 += 더한 요소의 값    

while (!(pq.size() ==1)) {
    int a = pq.poll();
    int b = pq.poll();
    sum += (a + b);
    pq.add(a + b);
}

 

예시

                                - (50 + 60) + (70 + 80) + ( (50 + 60) + (70 + 80) )

                                    - 50, 60을 비교    -> 결과 값 = 110 

                                                                                    +

                                    - 70, 80을 비교    -> 결과 값 = 150

                                                                                    +

                                    - 110, 150을 비교 -> 결과 값 = 260

                                                                                    = 520

 

 

전체 코드

import java.io.*;
import java.util.*;

public class Main {

    static int N;

    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        
        StringTokenizer st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken());
        
        PriorityQueue<Integer> pq = new PriorityQueue<Integer>();
        for (int i = 0; i < N; i++) {
            st = new StringTokenizer(br.readLine());
            pq.add(Integer.parseInt(st.nextToken()));
        }
        int sum = 0;

        // 초기 요소의 갯수가 1개라면 비교 횟수가 존재 X
        // 누적 합 과정 중 pq의 갯수가 한개가 남으면 비교가 끝남
        while ((pq.size() > 1)) {
            int a = pq.poll();
            int b = pq.poll();
            sum += (a + b);
            pq.add(a + b);
        }
        
        System.out.println(sum);
    }
}