알고리즘

백준 - 17825 주사위 윷놀이 (Java)

ksb-dev 2023. 7. 25. 20:20

구현 문제입니다.

 

문제 조건 자체는 이해하기 어려운 부분은 딱히 없는거 같습니다.

하지만, 이것을 코드로 구현하려니 마땅한 아이디어가 떠오르지 않았습니다.

3~4일간 이 문제만 붙잡고 백준 질문게시판에서 얻은 테스트 케이스를 직접 써보면서 경우의 수를 확인하고,

수많은 디버깅을 한 끝에 맞출 수 있었습니다.

제가 문제를 해결한 방법을 아래 글을 통해 공유하겠습니다.

 

1. 그래프로 윷놀이 판 구현하기

문제에서 주어지는 주사위 윷놀이 판의 모습은 아래 그림과 같습니다.

이 모습을 보자마자 그래프를 떠올렸고, 그림을 그래프로 표현하고자 했습니다.

우선 파란색으로 갈 수 있는 곳을 지름길(Shortcut)이라 가정했습니다.

💡 [10, 20, 30에서 출발해 40까지의 길이 모두 지름길입니다.]

💡 [편의상 빨간색으로만 갈 수 있는 길을 둘레길이라 하겠습니다.]

 

또, 특정 숫자는 두 가지 길이 있습니다.

예를 들어, 22번은 20에서 시작할 경우 지름길로 가 24 수도 있지만,

지름길이 아닌 빨간색 경로를 따라 또 다른 24로 갈 수 있습니다.

 

다행히 이렇게 겹치는 번호는 지름길이 유무이거나 앞 뒤로 연결된 번호가 다르다는 것입니다.

따라서 지름길 또는 앞, 뒤로 연결된 정보로 움직임을 확정지을 수 있습니다.

private static class Node {
    int before; // 뒷 번호
    int from; // 현재 번호
    int to; // 앞 번호
    boolean isShortCut; // 지름길

    Node(int before, int from, int to, boolean isShortcut) {
        this.before = before;
        this.from = from;
        this.to = to;
        this.isShortCut = isShortcut;
    }
    ...
}
private static final List<List<Node>> graph = new ArrayList<>();

// 게임 판 생성
private static void makeGraph() {
    // 코드가 너무 길어 본 코드에서 확인하세요
}

 

 

저의 경우 빨간색 둘레길을 먼저 생성하여 그래프의 첫 번째에 두고,

파란색 지름길을 그래프의 두 번째에 두었습니다.

 

여기서 예상치 못한 문제를 발견했습니다.

바로 30번입니다.

아래 그림과 같이 30번만 세 가지 길이 존재합니다.

 

 

위 그림에 의해 제가 만든 그래프는 30번의 경우 아래와 같이 저장됩니다.

graph.get(30).get(0) // 28 -> 30 -> 32
graph.get(30).get(1) // 28 -> 30 -> 28
graph.get(30).get(2) // 25 -> 30 -> 35

 

30번의 0과 1번째는 이전의 값이 28로 똑같기 때문에 ,파란색 지름길인지 혹은 다음 번호로 확인해야 합니다.

30번의 2번째는 파란색 지름길이기 때문에, 이전 번호 혹은 다음 번호로 확인해야 합니다.

💡 [위에서 10, 20, 30에서 시작해 40까지 갈 수 있는 모든 경로를 파란색 지름길로 상정한다 했습니다.]

 

2. 말판 움직이기

말판은 총 네개가 존재합니다.

또, 입력으로 주사위가 나올 수 10개가 주어집니다.

 

네 개의 말판을 길이가 10인 중복 순열로 만들고,

각 말판이 만들어진 순열에 따라 움직이면 됩니다.

 

주의점은 크게 두 가지가 있습니다.

  1. 만들어진 순열에 따라 움직이기 때문에, 이미 도착 지점에 도착한 말판은 움직일 수 없다.
  2. 문제 조건에 의해 이동을 완료한 지점에 다른 말판이 존재하면 안된다.

이 두 가지 주의점을 만족하지 못하면 불가능한 방법이므로,

과감하게 해당 중복 순열로 만들어진 방법은 무시하시면 됩니다.

 

3. 문제 접근 순서

  1. graph로 윷놀이 판 생성
  2. 말판의 중복 순열 생성
  3. 중복 순열에 따른 말판 이동

 

[구현 코드]

import java.io.*;
import java.util.*;

public class Main {

    private static final int N = 10;
    private static final int M = 4;
    private static final int START = 0;
    private static final int END = 42;

    private static final int[] moves = new int[N];
    private static final List<List<Node>> graph = new ArrayList<>();

    private static int max = 0;

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

        for (int i = 0; i < N; i++) {
            moves[i] = Integer.parseInt(stz.nextToken());
        }

        makeGraph();

        int[] order = new int[N];
        makeOrder(order, 0); // 중복 순열

        System.out.println(max);

        br.close();
    }

    // 게임 판 생성
    private static void makeGraph() {
        graph.add(new ArrayList<>()); // 0
        graph.add(new ArrayList<>()); // 1

        // 빨간색길 먼저 생성
        graph.get(START).add(new Node(START, START, START + 2, false));
        for (int i = START + 2; i <= END; i++) {
            graph.add(new ArrayList<>());

            if (i % 2 == 0) {
                graph.get(i).add(new Node(i - 2, i, i + 2, false));
            }
        }

        // 지름길 생성
        // 10 ~ 25
        graph.get(10).add(new Node(8, 10, 13, true));
        for (int i = 13; i <= 16; i += 3) {
            graph.get(i).add(new Node(i - 3, i, i + 3, true));
        }
        graph.get(19).add(new Node(16, 19, 25, true));

        // 20 ~ 25
        for (int i = 20; i <= 22; i += 2) {
            graph.get(i).add(new Node(i - 2, i, i + 2, true));
        }
        graph.get(24).add(new Node(22, 24, 25, true));

        // 30 ~ 25
        graph.get(30).add(new Node(28, 30, 28, true));
        for (int i = 27; i >= 26; i--) {
            graph.get(i).add(new Node(i + 1, i, i - 1, true));
        }
        graph.get(28).add(new Node(30, 28, 27, true));

        // 25 ~ 40
        // 25로 가는 경우는 19, 24, 26이 있지만 결국 25에서 한 쪽 방으로만 움직이므로 -1로 함
        graph.get(25).add(new Node(-1, 25, 30, true));
        for (int i = 30; i <= 35; i += 5) {
            graph.get(i).add(new Node(i - 5, i, i + 5, true));
        }
    }

    // 중복 순열 생성
    private static void makeOrder(int[] order, int depth) {
        if (depth == N) {
            max = Math.max(max, moving(order));
            return;
        }

        for (int i = 0; i < M; i++) {
            order[depth] = i;
            makeOrder(order, depth + 1);
        }
    }

    // 만들어진 순열에 맞춰 윷놀이 이동
    private static int moving(int[] order) {
        int total = 0;
        Node[] locations = new Node[M];
        for (int i = 0; i < M; i++) {
            locations[i] = new Node(START, START, START + 2, false);
        }

        Loop1: for (int t = 0; t < N; t++) {
            int mal = order[t];
            int power = moves[t];

            Node currentNode = locations[mal];

            // 도착지점에 있는 것을 움직이려고 할 경우 -> 불가능함(만들어진 순열에 따라서만 움직여야 됨)
            if(currentNode.from == END){
                total = 0;
                break;
            }

            // 주사위 결과에 따른 이동지점
            Node nextNode = nextLocation(currentNode, power);

            // 이미 다른 말판이 이동하고자 하는 자리에 있을 경우
            for (int i = 0; i < 4; i++) {
                // 문제 조건에 의해 이동지점이 도착인 경우 가능
                if(locations[i].from == END) continue;

                // 이동을 마치는 칸에 다른 말이 있으면
                if ((locations[i].isEqualNode(nextNode))) {
                    total = 0;
                    break Loop1;
                }
            }

            locations[mal] = nextNode;
            total += nextNode.from;

            if(nextNode.from == END) total -= END;
        }

        return total;
    }

    private static Node nextLocation(Node currenNode, int power) {
        Node cn = currenNode;

        while (power-- > 0) {
            if (cn.from == END) break;

            if (graph.get(cn.to).size() == 2) {
                cn = (!cn.isShortCut) ? graph.get(cn.to).get(0) : graph.get(cn.to).get(1);

            } else if (graph.get(cn.to).size() == 1) {
                cn = graph.get(cn.to).get(0);
            } else {
                // 30일 때
                if (!cn.isShortCut) {
                    cn = graph.get(cn.to).get(0);
                } else {
                    if (cn.before == -1) {
                        cn = graph.get(cn.to).get(2); // 25 -> 30 일 경우
                    } else {
                        cn = graph.get(cn.to).get(1); // 28 -> 30 일 경우
                    }
                }
            }
        }

        if (cn.from == 10) {
            cn = graph.get(10).get(1);
        } else if (cn.from == 20) {
            cn = graph.get(20).get(1);
        } else if (cn.from == 30) {
            if(cn.to != 35) {
                cn = graph.get(30).get(1);
            }
        }

        return cn;
    }

    private static class Node {
        int before; // 뒷 번호
        int from; // 현재 번호
        int to; // 앞 번호
        boolean isShortCut; // 지름길

        Node(int before, int from, int to, boolean isShortcut) {
            this.before = before;
            this.from = from;
            this.to = to;
            this.isShortCut = isShortcut;
        }


        public boolean isEqualNode(Node node) {
            if(this.before != node.before || this.from != node.from ||
                    this.to != node.to || this.isShortCut != node.isShortCut) {
                return false;
            }
            return true;
        }
    }
}