Algorithm/Baekjoon

[백준 10090 : JAVA] Counting Inversions / 세그먼트 트리

팡트루야 2021. 3. 25. 22:47

문제


 

 

 

풀이


Inversion 이란 $A[i]>A[j]$ (단 $i<j$)인 경우를 말한다. 즉, 수열이 오름차순이나 내림차순으로 되어있지 않고, 크기가 역전되어 있는 것을 말한다. Inversion 갯수를 세기 위해 단순 브루트 포스를 이용하면 $O(n^{2})$인데 반해, 세그먼트 트리를 이용하면 $O(nlogn)$으로 구할 수 있다.

 

수열 [4, 2, 7, 1, 5, 6, 3] 에 대해 세그먼트 트리를 그려보면 다음과 같다.

 

[그림 1] 세그먼트 트리

 

구간을 담당하는 각 노드에 어떤 값을 담아야할지가 포인트인데, 수열에서 현재 가리키고 있는 값이 X라 할때, X 이전까지의 수열에서 X보다 작은 수의 갯수를 저장한다. X보다 작은 수는 X-1개가 존재한다. X보다 먼저 나온 수들 중 X보다 작은 수의 갯수를 K개라고 한다면 Inversion countion은 X-1-K개가 된다. 예를 들어, [1, 2, 3, 5, 6, 4, 7] 이라는 수열이 있고 현재 가리키는 X가 4라고 가정하자. 4보다 작은 수는 {1, 2, 3} 세 개가 있고, 수열에서 4 이전까지의 수열을 보면 [1, 2, 3, 5] 로 작은 수를 모두 포함하기 때문에 해당 X는 크기가 역전되지 않았다. 참고로, 5랑 역전되지 않았냐고 묻는다면 X가 5를 가리킬 때를 생각해보면 이미 5에서 카운팅되었다는 것을 알 수 있다. 

 

예를 들어, 위 수열에서 현재 가리키고 있는 값이 1이라고 할때, 세그먼트 트리에 저장되는 값은 다음과 같다.

 

[그림 2] 수열에서 현재 가리키는 값이 1일때 각 노드에 담기는 값

 

수열에서 1 이전까지의 수열은 [4, 2, 7] 이다. 

 

 

 

코드


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

// Counting Inversions
public class Main {

    private static int N;
    private static long result;
    private static int[] segmentTree;

    public static void main(String[] args) throws Exception {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        N = Integer.parseInt(br.readLine());
        segmentTree = new int[N * 4];

        // 세그먼트 트리에서 각 노드의 값: 1~X 구간의 값들 중 지금까지 나온 값들의 갯수
        // 예를 들어, [4, 2, 7, 1, 5, 6, 3] 수열에서 현재 가리키는 값이 1일 때, 1~7 구간을 담당하는 1번 노드의 값은 3이다.
        // 현재 가리키는 1 이전의 수들 중 1~7 구간에 존재하는 값이 총 3개이기 때문이다.

        // 내가 현재 보는 수가 X라면 X보다 작은 수의 갯수는 X-1개다.
        // 먼저 나온 수들 중 현재 보고 있는 X보다 작은 수의 갯수가 a개라면 X에 의해 결정되는 경우의 수는 X-1-a 이다.
        // 따라서, 세그먼트 트리를 이용하여 수열을 받을 때마다 X보다 작은 수의 갯수를 구해주어 X-1-a를 결과값에 더해준 후,
        // 세그먼트 트리에서 X의 위치에 +1을 업데이트한다.
        StringTokenizer st = new StringTokenizer(br.readLine());
        for (int i = 1; i <= N; i++) {
            int X = Integer.parseInt(st.nextToken());
            result += X - 1 - query(1, N, 1, 1, X - 1);
            update(1, N, 1, X);
        }

        System.out.println(result);
    }

    // [start, end] : 노드가 담당하는 구간
    private static int update(int start, int end, int node, int pos) {
        if (pos < start || end < pos) return segmentTree[node];
        if (start == end) return segmentTree[node] += 1;
        int mid = (start + end) >> 1;
        return segmentTree[node] = update(start, mid, node * 2, pos) + update(mid + 1, end, node * 2 + 1, pos);
    }

    // [left, right] : 구하고자하는 구간
    private static int query(int start, int end, int node, int left, int right) {
        if (right < start || end < left) return 0;
        if (left <= start && end <= right) return segmentTree[node];
        int mid = (start + end) >> 1;
        return query(start, mid, node * 2, left, right) + query(mid + 1, end, node * 2 + 1, left, right);
    }
}