코딩 테스트/백준

백준 11505 자바 - 구간 곱 구하기

www.acmicpc.net/problem/11505

 

11505번: 구간 곱 구하기

첫째 줄에 수의 개수 N(1 ≤ N ≤ 1,000,000)과 M(1 ≤ M ≤ 10,000), K(1 ≤ K ≤ 10,000) 가 주어진다. M은 수의 변경이 일어나는 횟수이고, K는 구간의 곱을 구하는 횟수이다. 그리고 둘째 줄부터 N+1번째 줄

www.acmicpc.net

세그먼트를 이용해서 구간의 곱을 구하면 된다.

문제에서 신경쓸 부분으로는

1. 입력받은 값은 항상 1,000,000,007보다 낮으므로 항상 나머지가 된다.

2. 입력받은 값은 최대 1,000,000 이므로 곱을 누적하면 숫자가 커지므로 tree에 저장할때는 나머지값으로 셋팅한다.

3. 두수 곱의 나머지는 두수의 나머지의 곱과 같다(단 나누는 수보다 큰경우는 나머지를 나눠야 함) 

 

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

// 구간 곱 구하기 문제 - 11505번
public class PartMum {

    static int N, M, K;
    static int[] arr;
    static long[] tree;
    static int d = 1000000007;

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

        st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken());
        M = Integer.parseInt(st.nextToken());
        K = Integer.parseInt(st.nextToken());

        arr = new int[N];
        tree = new long[N * 4];

        for(int i = 0; i < N; i++) {
            arr[i] = Integer.parseInt(br.readLine());
        }

        init(0, arr.length -1, 1);

        for(int i = 0; i < M + K; i++) {
            st = new StringTokenizer(br.readLine());
            int a = Integer.parseInt(st.nextToken());
            int b = Integer.parseInt(st.nextToken());
            int c = Integer.parseInt(st.nextToken());

            // 변경
            if(a == 1) {
                update(0, arr.length - 1, 1, b-1, c);
            }
            // 곱하여 출력
            else {
                System.out.println((segment(0, arr.length - 1, 1, b-1, c-1)));
            }


        }
    }

    static long init(int start, int end, int node) {
        if(start == end) {
            return tree[node] = arr[start];
        }

        int mid = (start + end) / 2;
        return tree[node] = init(start, mid, node * 2) *
                init(mid + 1, end, node * 2 + 1) % d;
    }

    static long update(int start, int end, int node, int index, int c ) {
        // index가 범위 밖인 경우 tree의 값을 수정하지 않고 바로 리턴
        if(index < start || index > end) {
            return tree[node];
        }

        // 같으면 leaf 노드 이므로 변경 입력값으 1000000이하 이고 나누는 수는 10억이기 때문에 입력받은 수 그대로 나머지가 됨
        if(start == end) {
            return tree[node] = c;
        }

        int mid = (start + end) / 2;
        return tree[node] = update(start, mid, node * 2, index, c) *
                update(mid+1, end, node * 2 + 1, index, c) % d;
    }

    static long segment(int start, int end, int node, int left, int right) {
        // 범위 밖인 경우 - 곱하기이기때문에 0 이 아닌 1을 리턴
        if(left > end || right < start) {
            return 1;
        }

        if(left <= start && right >= end ) {
            return tree[node];
        }

        int mid = (start + end) / 2;
        return  segment(start, mid, node * 2, left, right) *
                segment(mid + 1, end, node * 2 + 1, left, right) % d;
    }
}