본문 바로가기
Algorithm/Baekjoon

Baekjoon 2116 주사위 쌓기 JAVA

by Hunveloper 2022. 2. 16.
728x90
 

2116번: 주사위 쌓기

첫줄에는 주사위의 개수가 입력된다. 그 다음 줄부터는 한 줄에 하나씩 주사위의 종류가 1번 주사위부터 주사위 번호 순서대로 입력된다. 주사위의 종류는 각 면에 적혀진 숫자가 그림1에 있는

www.acmicpc.net

문제

천수는 여러 종류의 주사위를 가지고 쌓기 놀이를 하고 있다. 주사위의 모양은 모두 크기가 같은 정육면체이며 각 면에는 1부터 6까지의 숫자가 하나씩 적혀있다. 그러나 보통 주사위처럼 마주 보는 면에 적혀진 숫자의 합이 반드시 7이 되는 것은 아니다.

주사위 쌓기 놀이는 아래에서부터 1번 주사위, 2번 주사위, 3번 주사위, … 의 순서로 쌓는 것이다. 쌓을 때 다음과 같은 규칙을 지켜야 한다: 서로 붙어 있는 두 개의 주사위에서 아래에 있는 주사위의 윗면에 적혀있는 숫자는 위에 있는 주사위의 아랫면에 적혀있는 숫자와 같아야 한다. 다시 말해서, 1번 주사위 윗면의 숫자는 2번 주사위 아랫면의 숫자와 같고, 2번 주사위 윗면의 숫자는 3번 주사위 아랫면의 숫자와 같아야 한다. 단, 1번 주사위는 마음대로 놓을 수 있다.

이렇게 쌓아 놓으면 긴 사각 기둥이 된다. 이 사각 기둥에는 4개의 긴 옆면이 있다. 이 4개의 옆면 중에서 어느 한 면의 숫자의 합이 최대가 되도록 주사위를 쌓고자 한다. 이렇게 하기 위하여 각 주사위를 위 아래를 고정한 채 옆으로 90도, 180도, 또는 270도 돌릴 수 있다. 한 옆면의 숫자의 합의 최댓값을 구하는 프로그램을 작성하시오.

입력

첫줄에는 주사위의 개수가 입력된다. 그 다음 줄부터는 한 줄에 하나씩 주사위의 종류가 1번 주사위부터 주사위 번호 순서대로 입력된다. 주사위의 종류는 각 면에 적혀진 숫자가 그림1에 있는 주사위의 전개도에서 A, B, C, D, E, F 의 순서로 입력된다. 입력되는 숫자 사이에는 빈 칸이 하나씩 있다. 주사위의 개수는 10,000개 이하이며 종류가 같은 주사위도 있을 수 있다.

 

출력

첫줄에 한 옆면의 숫자의 합이 가장 큰 값을 출력한다.

풀이

HashMap을 이용하여 반대편 주사위의 번호를 매칭시킴

main에서 함수를 호출할때 첫번째 주사위의 상태 6개를 다 검사함

함수안에서 다음 주사위의 마주보고 있는 면을 찾고 그 다음 주사위와 비교하면서 다음에 밑으로 올 index를 저장

현재 바닥과 윗부분을 빼고 가장 큰 수를 찾음

다음 주사위를 탐색

코드
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.HashMap;
import java.util.StringTokenizer;

public class Main {
	static HashMap<Integer, Integer> match = new HashMap<Integer, Integer>();

	public static int maxnum(int[][] dice, int cnt, int k, int n) {
		if(cnt==n)
			return 0;
		
		int nk=-1;	// 다음 주사위의 마주보고 있는 면
		for(int i=0;i<6;i++)
			if(dice[cnt+1][i]==dice[cnt][match.get(k)])	// 현재주사위의 반대면을 찾고, 다음 주사위의 인덱스를 찾아서 nk에 저장
				nk=i;
		
		boolean[] chk = new boolean[6];	// 현재 밑면과 윗면을 boolean으로 처리
		int tmax = 0;		
		if (k == 0 || k == 5)
			chk[0] = chk[5] = true;
		else if (k == 1 || k == 3)
			chk[1] = chk[3] = true;
		else if (k == 2 || k == 4)
			chk[2] = chk[4] = true;
		for (int i = 0; i < 6; i++) 	// 남은 값들 중 최대값을 현재 cnt위치의 최대값으로 지정
			if (!chk[i])
				tmax = Math.max(tmax, dice[cnt][i]);
		tmax += maxnum(dice, cnt + 1, nk, n);	// 현재 값에 다음 주사위의 값을 찾아서 더함
		return tmax;	// 더해진 값을 반환
	}

	public static void main(String[] args) throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		int n = Integer.parseInt(br.readLine());
		int[][] dice = new int[n+1][6];
		for (int i = 0; i < n; i++) {
			StringTokenizer st = new StringTokenizer(br.readLine());
			for (int j = 0; j < 6; j++)
				dice[i][j] = Integer.parseInt(st.nextToken());
		}
		// 주사위의 반대면을 알기 위한 Dict
		match.put(0, 5);
		match.put(1, 3);
		match.put(2, 4);
		match.put(5, 0);
		match.put(3, 1);
		match.put(4, 2);
		
		int max = 0;
		for (int i = 0; i < 6; i++) {
			int temp = maxnum(dice, 0, i, n);	// 첫번째 주사위가 놓일수 있는 모든 경우를 탐색
			if (max < temp)
				max = temp;
		}
		System.out.println(max);
	}
}
728x90
728x90

'Algorithm > Baekjoon' 카테고리의 다른 글

Baekjoon 17219 비밀번호 찾기 JAVA  (0) 2022.02.16
Baekjoon 11286 절댓값 힙 JAVA  (0) 2022.02.16
Baekjoon 14696 딱지놀이 JAVA  (0) 2022.02.16
Baekjoon 13300 방 배정 JAVA  (0) 2022.02.16
Baekjoon 10163 색종이 JAVA  (0) 2022.02.16

댓글