Algorithm/Baekjoon

[백준 3653 : JAVA] 영화 수집 / 세그먼트 트리

팡트루야 2021. 3. 30. 21:42

문제


 

 

 

풀이


세그먼트 트리 응용문제로, 창의성을 요구하는 문제다.

 

[그림 1] 구간으로 설정해야할 것

 

한 번 보고난 영화 DVD는 위로 옮겨져야 하기 때문에 옮겨질(m)만큼의 크기도 추가해 배열을 할당한다.

 

3번 영화를 보고난 후에는 3번 영화의 인덱스를 변경해줘야 하기 때문에, 각 [영화 번호, 인덱스] 쌍을 알고 있어야 한다.

영화번호-인덱스 쌍을 저장하기 위해 Map(=Dictionary) 자료구조를 이용하는 대신, 좀 더 원시적인 방법인 배열을 이용했다.

for (int i = 1; i <= n; i++) {
    moviePosInStack[i] = m + i - 1; // 배열[영화번호] = 인덱스 
}

 

 

 

코드


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

public class Main {

    private static int n, m;
    private static SegmentTree tree;

    public static void main(String[] args) throws Exception {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringBuilder sb = new StringBuilder();

        int T = Integer.parseInt(br.readLine());
        while (T-- > 0) {
            StringTokenizer st = new StringTokenizer(br.readLine());
            n = Integer.parseInt(st.nextToken());
            m = Integer.parseInt(st.nextToken());

            int[] movieStack = new int[n + m];
            int[] moviePosInStack = new int[n + 1];
            for (int i = 1; i <= n; i++) {
                movieStack[m + i - 1] = 1;
                moviePosInStack[i] = m + i - 1; // [영화번호, 인덱스] 쌍을 담는다.
            }

            tree = new SegmentTree(movieStack, n + m);
            tree.init(1, 0, n + m - 1);

            st = new StringTokenizer(br.readLine());
            for (int i = 0; i < m; i++) {
                int selectedMovie = Integer.parseInt(st.nextToken());

                // Query
                int result = tree.query(0, moviePosInStack[selectedMovie] - 1, 1, 0, n + m - 1);

                // Update
                tree.update(moviePosInStack[selectedMovie], 0, 1, 0, n + m - 1);
                moviePosInStack[selectedMovie] = (m - 1) - i; // 해당 DVD의 위치(인덱스)를 가장 위로 옮긴다.
                tree.update(moviePosInStack[selectedMovie], 1, 1, 0, n + m - 1);

                sb.append(result).append(" ");
            }

            sb.append("\n");
        }

        System.out.println(sb.toString());
    }

    private static class SegmentTree {

        int[] arr;
        int[] tree;

        public SegmentTree(int[] arr, int n) {
            this.arr = arr;
            int k = (int) Math.ceil(Math.log(n) / Math.log(2));
            int height = k + 1;
            int size = (int) Math.pow(2, height);
            tree = new int[size];
        }

        public int init(int node, int start, int end) {
            if (start == end) {
                return tree[node] = arr[start];
            }
            int mid = (start + end) / 2;
            return tree[node] = init(node * 2, start, mid) + init(node * 2 + 1, mid + 1, end);
        }

        public int update(int idx, int newValue, int node, int start, int end) {
            if (idx < start || end < idx) return tree[node];

            if (start == end) return tree[node] = newValue;

            int mid = (start + end) / 2;
            return tree[node] = update(idx, newValue, node * 2, start, mid) + update(idx, newValue, node * 2 + 1, mid + 1, end);
        }

        public int query(int left, int right, int node, int start, int end) {
            if (right < start || end < left) return 0;
            if (left <= start && end <= right) return tree[node];
            int mid = (start + end) / 2;
            return tree[node] = query(left, right, node * 2, start, mid) + query(left, right, node * 2 + 1, mid + 1, end);
        }
    }
}