최솟값

Huisu·2024년 7월 16일
0

Coding Test Practice

목록 보기
99/119
post-thumbnail

문제

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

문제 설명

N(1 ≤ N ≤ 100,000)개의 정수들이 있을 때, a번째 정수부터 b번째 정수까지 중에서 제일 작은 정수를 찾는 것은 어려운 일이 아니다. 하지만 이와 같은 a, b의 쌍이 M(1 ≤ M ≤ 100,000)개 주어졌을 때는 어려운 문제가 된다. 이 문제를 해결해 보자.

여기서 a번째라는 것은 입력되는 순서로 a번째라는 이야기이다. 예를 들어 a=1, b=3이라면 입력된 순서대로 1번, 2번, 3번 정수 중에서 최솟값을 찾아야 한다. 각각의 정수들은 1이상 1,000,000,000이하의 값을 갖는다.

제한 사항

첫째 줄에 N, M이 주어진다. 다음 N개의 줄에는 N개의 정수가 주어진다. 다음 M개의 줄에는 a, b의 쌍이 주어진다.

M개의 줄에 입력받은 순서대로 각 a, b에 대한 답을 출력한다.

입출력 예

입력출력
10 4
75
30
100
38
50
51
52
20
81
5
1 105
3 538
6 920
8 105

제출 코드

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.StringTokenizer;

// https://www.acmicpc.net/problem/10868
public class one10868 {
    public void solution() throws IOException {
        BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer infoToken = new StringTokenizer(reader.readLine());
        int n = Integer.parseInt(infoToken.nextToken());
        int m = Integer.parseInt(infoToken.nextToken());

        long[] arr = new long[n];
        for (int i = 0; i < n; i++) {
            arr[i] = Long.parseLong(reader.readLine());
        }

        int h = (int) Math.ceil(Math.log(n) / Math.log(2));
        int size = (1 << (h + 1));
        long[] tree = new long[size];

        init(arr, tree, 1, 0, n - 1);

        for (int i = 0; i < m; i++) {
            infoToken = new StringTokenizer(reader.readLine());
            int left = Integer.parseInt(infoToken.nextToken()) - 1;
            int right = Integer.parseInt(infoToken.nextToken()) - 1;
            System.out.println(query(tree, 1, 0, n - 1, left, right));
        }
    }

    private long query(long[] tree, int node, int start, int end, int left, int right) {
        if (right < start || left > end) {
            return Long.MAX_VALUE;
        } else if (left <= start && end <= right) {
            return tree[node];
        } else {
            long leftMin = query(tree, node * 2, start, (start + end) / 2, left, right);
            long rightMin = query(tree, node * 2 + 1, (start + end) / 2 + 1, end, left, right);
            return Math.min(leftMin, rightMin);
        }
    }

    private void init(long[] arr, long[] tree, int node, int start, int end) {
        if (start == end) {
            tree[node] = arr[start];
        } else {
            init(arr, tree, node * 2, start, (start + end) / 2);
            init(arr, tree, node * 2 + 1, (start + end) / 2 + 1, end);
            tree[node] = Math.min(tree[node * 2], tree[node * 2 + 1]);
        }
    }

    public static void main(String[] args) throws IOException {
        new one10868().solution();
    }
}

0개의 댓글