Algorithm/Baekjoon
[백준 17825 : JAVA] 주사위 윷놀이 / 백트래킹 + 시뮬레이션
팡트루야
2020. 11. 22. 22:36
문제
풀이
개인적으로 많이 어려웠던 문제.
해당 문제에서 요구하는 요구사항은 크게 2가지다.
- 주어진 맵(윷놀이판)을 어떻게 표현할 것인가?
- 10개의 턴에 대한 4개의 말 순서 배치는 어떻게할 것인가? (순열 --- 백트래킹 알고리즘 이용)
먼저 첫 번째 요구사항을 보자.
주어진 윷놀이판을 1차원으로 보기 쉽게끔 나타내면 다음과 같다.
위의 1차원 구조에서 포인트는 10, 20, 30은 두 가지 방향을 가진다는 것이다.
그럼 노드가 가져야할 정보는 다음과 같다.
해당 칸의 점수, 해당 칸이 비었는지(다른 말이 있는지), 지름길(파란 방향)이 있는지, 도착점인지
배열에 담아도 되지만, 그보다는 연결 리스트로 담는게 좀 더 직관적이라 생각한다.
( 개인적으로 아래와 같이 연결 리스트를 만들어 사용해보는건 처음이었는데, 많은 도움이 됬다. )
static class Node { // 윷놀이 판을 연결 리스트 구조에 담는다.
int score; // 윷놀이판에서 해당 칸의 점수.
boolean isEmpty; // 해당 칸이 비었는지. (다른 말이 와있는 칸인지 체크)
boolean isFinish; // 도착 지점인지를 체크.
Node next;
Node fastPath; // 윷놀이판에서 10, 20, 30은 두 가지 방향이 존재한다.
public Node(int score) {
this.score = score;
this.isEmpty = true;
}
// 노드 연결 (연결 리스트 구조)
public Node addNext(int score) {
Node nextNode = new Node(score);
this.next = nextNode;
return nextNode;
}
// 노드 찾기 (지름길 놓는 지점을 찾기 위한 함수)
public static Node getNode(Node start, int target) {
Node temp = start;
while (true) { // 시작 지점부터 탐색해가며 특정 노드를 찾습니다.
if (temp == null) return null;
if (temp.score == target) return temp;
temp = temp.next;
}
}
}
두 번째 요구사항은 10개의 턴에 대한 4개의 말들의 순서 배치인데, 이는 백트래킹으로 간단히 구할 수 있다.
private static void permutation(int depth) {
if (depth >= 11) {
answer = Math.max(answer, gameStart());
return;
}
for (int i = 1; i <= 4; i++) {
order[depth] = i;
permutation(depth + 1);
}
}
코드
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.StringTokenizer;
public class Main {
private static BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
private static int[] dice, order; // 주사위, 배치순서
private static Node[] horse; // 4개의 말
private static Node start; // 시작지점
private static int answer = Integer.MIN_VALUE;
static class Node { // 윷놀이 판을 연결 리스트 구조에 담는다.
int score; // 윷놀이판에서 해당 칸의 점수.
boolean isEmpty; // 해당 칸이 비었는지. (다른 말이 와있는 칸인지 체크)
boolean isFinish; // 도착 지점인지를 체크.
Node next;
Node fastPath; // 윷놀이판에서 10, 20, 30은 두 가지 방향이 존재한다.
public Node(int score) {
this.score = score;
this.isEmpty = true;
}
// 노드 연결 (연결 리스트 구조)
public Node addNext(int score) {
Node nextNode = new Node(score);
this.next = nextNode;
return nextNode;
}
// 노드 찾기 (지름길 놓는 지점을 찾기 위한 함수)
public static Node getNode(Node start, int target) {
Node temp = start;
while (true) { // 시작 지점부터 탐색해가며 특정 노드를 찾습니다.
if (temp == null) return null;
if (temp.score == target) return temp;
temp = temp.next;
}
}
}
public static void main(String[] args) throws IOException {
StringTokenizer st = new StringTokenizer(br.readLine());
dice = new int[10 + 1];
order = new int[10 + 1];
horse = new Node[4 + 1];
for (int i = 1; i <= 10; i++) {
dice[i] = Integer.parseInt(st.nextToken());
}
init(); // 윷놀이판 설정
permutation(1); // 10개의 주사위 결과를 말들에게 할당
System.out.println(answer);
}
private static void permutation(int depth) {
if (depth >= 11) {
answer = Math.max(answer, gameStart());
return;
}
for (int i = 1; i <= 4; i++) {
order[depth] = i;
permutation(depth + 1);
}
}
private static int gameStart() {
Arrays.fill(horse, start); // 말들을 시작 지점으로 배치
int score = 0;
for (int i = 1; i <= 10; i++) {
Node cur = horse[order[i]]; // 순열로 할당된 순서대로 말들을 움직입니다.
cur.isEmpty = true; // 현재 있는 칸을 비워줍니다.
for (int j = 1; j <= dice[i]; j++) { // 주사위에 나온 수치만큼 이동합니다.
if (j == 1 && cur.fastPath != null) { // 처음 이동을 시작하려는 위치가 파란색 칸이라면.
cur = cur.fastPath; // 지름길로 이동(파란색 방향)
} else {
cur = cur.next; // 빨간색 방향으로 이동
}
}
horse[order[i]] = cur; // 이동 후, 말 위치 설정
if (!cur.isEmpty && !cur.isFinish) { // 이동을 마친 칸에 다른 말이 있다면, 해당 말은 고를 수 없다.
score = 0; // 주사위에 할당받은 말들의 순서가 무효처리.
break;
} else {
cur.isEmpty = false; // 말이 존재하는 것으로 표시
score += cur.score; // 해당 칸의 점수 추가
}
} // 게임종료
// 다음 게임을 위해 말들의 위치를 지워줍니다.
for (int i = 1; i <= 4; i++)
horse[i].isEmpty = true;
return score; // 획든한 점수 반환
}
private static void init() {
start = new Node(0); // 시작 지점. (시작 지점의 스코어는 0)
Node temp = start;
for (int i = 2; i <= 40; i += 2) {
temp = temp.addNext(i); // 윷놀이판 바깥쪽 경로 설정
}
Node end = temp.addNext(0); // 도착 지점. (도착 지점의 스코어는 0)
end.isFinish = true;
end.next = end; // 자기자신을 가르키도록 설정 (도착 지점을 넘어서는 이동에 대해 NPE 방지)
Node crossroads = new Node(25); // 가운데 교차점 (score = 25)
// 교차점(25) -> 30 -> 35 -> 40
temp = crossroads.addNext(30);
temp = temp.addNext(35);
temp.next = Node.getNode(start, 40);
// 10 -> 13 -> 16 -> 19 -> 25(교차점)
temp = Node.getNode(start, 10);
temp = temp.fastPath = new Node(13);
temp = temp.addNext(16);
temp = temp.addNext(19);
temp.next = crossroads;
// 20 -> 22 -> 24 -> 25(교차점)
temp = Node.getNode(start, 20);
temp = temp.fastPath = new Node(22);
temp = temp.addNext(24);
temp.next = crossroads;
// 30 -> 28 -> 27 -> 26 -> 25(교차점)
temp = Node.getNode(start, 30);
temp = temp.fastPath = new Node(28);
temp = temp.addNext(27);
temp = temp.addNext(26);
temp.next = crossroads;
}
}
참고자료
[1] octorbirth.tistory.com/235