https://www.acmicpc.net/problem/2616
문제
기차는 맨 앞에 있는 기관차 1대가 손님이 탄 객차 여러 칸을 끌고 간다. 기관차가 고장나면 기차를 운행할 수 없게 되므로 최근 철도청은 기관차 고장에 대비하여 몇몇 역에 소형 기관차 3대를 배치하기로 결정하였다. 소형 기관차는 평소에 이용하는 기관차보다 훨씬 적은 수의 객차만을 끌 수 있다.
기관차가 고장났을 때 끌고 가던 객차 모두를 소형 기관차 3대가 나누어 끌 수 없기 때문에, 소형 기관차들이 어떤 객차들을 끌고 가는 것이 좋을까하는 문제를 고민하다가 다음과 같이 하기로 결정하였다.
- 소형 기관차가 최대로 끌 수 있는 객차의 수를 미리 정해 놓고, 그보다 많은 수의 객차를 절대로 끌게 하지 않는다. 3대의 소형 기관차가 최대로 끌 수 있는 객차의 수는 서로 같다.
- 소형 기관차 3대를 이용하여 최대한 많은 손님을 목적지까지 운송하도록 한다. 각 객차 마다 타고 있는 손님의 수는 미리 알고 있고, 다른 객차로 손님들이 이동하는 것은 허용하지 않는다.
- 각 소형 기관차는 번호가 연속적으로 이어진 객차를 끌게 한다. 객차는 기관차 바로 뒤에 있는 객차부터 시작하여 1번 부터 차례로 번호가 붙어있다.
예를 들어 기관차가 끌고 가던 객차가 7칸이고, 소형 기관차 1대가 최대로 끌 수 있는 객차 수는 2칸이라고 하자. 그리고 1번 부터 7번까지 각 객차에 타고 있는 손님의 수가 아래 표와 같다고 하자. 괄호속에 있는 숫자는 객차 번호를 나타낸다.
(1) | (2) | (3) | (4) | (5) | (6) | (7) |
35 | 40 | 50 | 10 | 30 | 45 | 60 |
소형 기관차 3대는 각각 1-2번, 3-4번, 그리고 6-7번 객차를 끌고 가면 손님 240명을 운송할 수 있고, 이보다 많은 수의 손님을 운송할 수 없다.
기관차가 끌고 가던 객차의 수와 각 객차에 타고 있던 손님의 수, 그리고 소형 기관차가 최대로 끌수 있는 객차의 수가 주어질 때, 소형 기관차 3대를 이용하여 최대로 운송할 수 있는 손님 수를 구하는 프로그램을 작성하시오.
입력
첫째 줄에 기관차가 끌고 가던 객차의 수가 입력된다. 그 수는 50,000 이하이다. 둘째 줄에는 기관차가 끌고 가던 객차에 타고 있는 손님의 수가 1번 객차부터 차례로 입력된다. 한 객차에 타고 있는 손님의 수는 100명 이하이고, 입력되는 숫자들 사이에 빈칸이 하나씩 있다. 셋째 줄에는 소형 기관차가 최대로 끌 수 있는 객차의 수가 입력된다. 그 수는 기관차가 끌고 가던 객차 수의 1/3보다 적다.
출력
한 줄에 소형 기관차 3대를 이용하여 최대로 운송할 수 있는 손님 수를 출력한다.
예제 입력 1
7
35 40 50 10 30 45 60
2
예제 출력 1
240
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.StringTokenizer;
public class Main {
public static void main(String[] args) throws Exception {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
StringTokenizer st;
int n = Integer.parseInt(br.readLine()); // 객차 수
int[][] arr = new int[5][n]; // 객차 별 손님의 수와 dp 계산을 넣을 배열
st = new StringTokenizer(br.readLine()); // 객차 별 손님 수 입력
for (int i = 0; i < n; i++) {
arr[0][i] = Integer.parseInt(st.nextToken());
}
int l = Integer.parseInt(br.readLine()); // 최대로 끌 수 있는 객차 수
int sum = 0;
for (int i = 0; i < n - l + 1; i++) {
if(i == 0) for (int j = 0; j < l; j++) sum += arr[0][j]; // 처음만 i부터 l개 합을 구하고
else sum = sum - arr[0][i - 1] + arr[0][i + l - 1]; // 다음부턴 앞에 더해주고 맨 뒤 빼줌
arr[1][i] = sum; // i~i+l-1까지의 합을 arr[1][i]에 담는다.
// 첫 번째 기관차는 i번째 객차까지 검사했을 때 선택할 수 있는 가장 많은 손님 수 저장
if(i == 0) arr[2][i] = arr[1][i];
else arr[2][i] = Math.max(arr[2][i - 1], arr[1][i]);
// 두 번째 기관차는 첫 번째 기관차가 선택할 수 있는 최대 손님 수 + 자신이 선택할 수 있는 최대 손님 수
if(i >= l) { // 이미 첫 번째 기관차가 선택한 이후이므로 인덱스가 l부터 시작해야 한다.
arr[3][i] = Math.max(arr[3][i - 1], arr[2][i - l] + arr[1][i]);
}
// 세 번째 기관차는 두 번째 기관차까지 선택할 수 있는 최대 손님 수 + 자신이 선택할 수 있는 최대 손님 수
if(i >= 2 * l) { // 이미 두 번째 기관차까지 선택한 이후이므로 인덱스가 2l부터 시작해야 한다.
arr[4][i] = Math.max(arr[4][i - 1], arr[3][i - l] + arr[1][i]);
}
}
// 세 번째 기관차의 마지막 경우의 수가 담긴 곳이 최대값
System.out.println(arr[4][n - l]);
}
}
'Problem Solving > BOJ' 카테고리의 다른 글
[백준, BOJ 7453] 합이 0인 네 정수 (java) (0) | 2023.02.28 |
---|---|
[백준, BOJ 17471] 게리맨더링(java) (0) | 2023.02.28 |
[백준, BOJ 9252] LCS 2 (java) (0) | 2023.02.27 |
[백준, BOJ 15486] 퇴사 2 (java) (0) | 2023.02.27 |
[백준, BOJ 15565] 귀여운 라이언 (java) (0) | 2023.02.27 |