ABOUT ME

-

Today
-
Yesterday
-
Total
-
  • [BOJ] 다리 만들기 2
    Algorithm/BOJ 2020. 9. 8. 19:46

    [17472] 다리 만들기 2

    www.acmicpc.net/problem/17472

     

    Solution

    • 다리 만들기 2 문제는 기존의 다리 만들기 문제와 세 가지 다른점이 있다.
    • 다리의 모양에 제한이 있는 것과 한 개의 다리가 아닌 모든 섬을 연결하는 다리를 만드는 것 그리고 다리의 길이는 최소 2 이상이어야 한다는 것이다.
    • 모든 섬을 연결하는 최소 비용의 다리를 만든다는 점에서 Spanning Tree idea를 떠올려야 한다.
    • 먼저 각각의 섬을 구분하기 위한 numbering 작업을 수행한다.
    • MST(최소 신장 트리)를 구하기 위해 크루스칼 알고리즘을 사용했다.
    • map의 모든 cell에 대해 DFS로 섬과 섬을 연결하는 비용(거리)을 구해 우선순위 큐에 넣는다.
    • 이때 주의할 점은 다리는 'ㅡ' 혹은 'l' 모양으로만 연결될 수 있기 때문에  DFS에서 방향 정보를 함께 전달하여 섬과 섬 사이의 거리를 구한다.
    • 마지막으로 거리 정보가 들어있는 우선순위 큐에 대해 union-find로 섬을 연결한다.
    • 우선순위 큐에서 간선을 뽑고 Cycle 여부를 검사하는 코드를 반복 수행하여 모든 섬을 연결한다.
    • Disjoint Set에서 부모 노드를 저장하는 parent 배열에서 루트가 하나 이상일 경우 -1을 반환한다.

     

     

    소스코드

     

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138
    139
    140
    141
    142
    143
    144
    145
    146
    147
    148
    149
    150
    151
    152
    153
    154
    155
    156
    157
    158
    159
    160
    161
    162
    163
    164
    165
    166
    167
    168
    169
    170
    171
    172
    173
    174
    175
    176
    177
    178
    179
    180
    181
    182
    183
    184
    185
    186
    187
    188
    189
    190
    import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.util.Arrays;
    import java.util.PriorityQueue;
    import java.util.StringTokenizer;
     
    public class Main {
        static int n, m;
        static int[][] map, visited;
        static int[] parent;
        static int[] dx = { 1-100 };
        static int[] dy = { 001-1 };
        static PriorityQueue<Node> pq;
     
        static class Node implements Comparable<Node> {
            private int start;
            private int end;
            private int len;
     
            public Node(int start, int end, int len) {
                this.start = start;
                this.end = end;
                this.len = len;
            }
     
            @Override
            public int compareTo(Node o) {
                return this.len - o.len;
            }
     
            @Override
            public String toString() {
                return "Node [start=" + start + ", end=" + end + ", len=" + len + "]";
            }
        }
     
        public static void main(String[] args) throws IOException {
            BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
            StringTokenizer st = new StringTokenizer(br.readLine());
            n = stoi(st.nextToken());
            m = stoi(st.nextToken());
     
            visited = new int[n][m];
            map = new int[n][m];
            pq = new PriorityQueue<>();
     
            // input
            for (int i = 0; i < n; i++) {
                st = new StringTokenizer(br.readLine());
                for (int j = 0; j < m; j++) {
                    if (stoi(st.nextToken()) == 1) {
                        map[i][j] = -1;
                    }
                }
            }
     
            // numbering
            int num = 1;
            for (int i = 0; i < n; i++) {
                for (int j = 0; j < m; j++) {
                    if (map[i][j] != 0 && visited[i][j] == 0) {
                        numbering(i, j, num++);
                    }
                }
            }
     
            // map의 모든 cell에 대해 4방향으로 DFS
            for (int i = 0; i < n; i++) {
                for (int j = 0; j < m; j++) {
                    for (int d = 0; d < 4; d++) {
                        if (map[i][j] != 0) {
                            init();
                            visited[i][j] = 1;
                            findLength(i, j, d, map[i][j]);
                        }
                    }
                }
            }
     
            parent = new int[num];
            for (int i = 1; i < num; i++) {
                parent[i] = -1;
            }
     
            System.out.println(kruskal(num));
        }
     
        static void numbering(int x, int y, int num) {
            if (map[x][y] == 0 || visited[x][y] == 1) {
                return;
            }
            map[x][y] = num;
            visited[x][y] = 1;
            int nx, ny;
            for (int i = 0; i < 4; i++) {
                nx = x + dx[i];
                ny = y + dy[i];
     
                if (nx < 0 || ny < 0 || nx >= n || ny >= m)
                    continue;
     
                if (map[nx][ny] == -1 && visited[nx][ny] == 0) {
                    numbering(nx, ny, num);
                }
            }
     
        }
     
        static void findLength(int x, int y, int dir, int label) {
            if (map[x][y] != 0 && map[x][y] != label) {
                pq.add(new Node(label, map[x][y], visited[x][y] - 2));
                return;
            }
     
            int nx = x + dx[dir];
            int ny = y + dy[dir];
     
            if (nx < 0 || ny < 0 || nx >= n || ny >= m)
                return;
     
            if (map[nx][ny] != label && visited[nx][ny] == 0) {
                visited[nx][ny] = visited[x][y] + 1;
                findLength(nx, ny, dir, label);
            }
        }
     
        static int kruskal(int num) {
            int result = 0;
            int count = 0;
            while (!pq.isEmpty() && count < num - 1) {
                Node node = pq.poll();
                // cycle 여부 확인 후 union
     
                if (node.len < 2)
                    continue;
     
                if (union(node.start, node.end)) {
                    result += node.len;
                    count++;
                }
            }
     
            count = 0;
            for (int i : parent) {
                if (i == -1) {
                    count++;
                }
            }
     
            return count == 1 ? result : -1;
        }
     
        static boolean union(int start, int end) {
            start = find(start);
            end = find(end);
            if (start != end) {
                parent[end] = start;
                return true;
            }
            return false;
        }
     
        static int find(int x) {
            if (parent[x] == -1) {
                return x;
            }
            return parent[x] = find(parent[x]);
        }
     
        static void init() {
            for (int i = 0; i < visited.length; i++) {
                Arrays.fill(visited[i], 0);
            }
        }
     
        static void print(int[][] arr) {
            for (int i = 0; i < arr.length; i++) {
                for (int j = 0; j < arr[i].length; j++) {
                    System.out.print(arr[i][j] + " ");
                }
                System.out.println();
            }
        }
     
        private static int stoi(String input) {
            return Integer.parseInt(input);
        }
    }
     
    cs

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

    [BOJ] 치즈  (0) 2020.10.19
    [BOJ] 후보 추천하기  (0) 2020.10.04
    [BOJ] 다리 만들기  (0) 2020.09.07
    [BOJ] 게리맨더링  (0) 2020.09.06
    [BOJ] 파이프 옮기기 1  (0) 2020.09.06

    댓글

Designed by Tistory.