Notice
Recent Posts
Recent Comments
Link
«   2025/01   »
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
Archives
Today
Total
관리 메뉴

최's 먹공로그

BOJ2146_다리만들기 본문

APS

BOJ2146_다리만들기

ChoiSH313 2019. 6. 9. 18:58

https://www.acmicpc.net/problem/2146


문제요약

1. 한 섬과 다른 섬을 잇는 다리 하나만을 만든다.

2. 한 섬과 다른 섬을 잇는 다리 중에서 가장 짧은 다리로 결정한다.

3. 지도가 주어질 때, 가장 짧은 다리 하나를 놓아 두 섬을 연결하는 방법을 찾자

4. 0은 바다, 1은 육지 , 항상 두 개 이상의 섬이 있는 데이터만 주어진다.

5. 가장 짧은 다리의 길이를 출력


문제issue

1. 다른 섬이라는 표시 어떤식으로??


해결흐름

1. bfs를 사용하여 섬을 분류한다. 1번 섬은 1로 맵핑 , 2번 섬은 2로 맵핑 ......

2. 각각의 섬에서 0과 인접한 곳을 발견하면 그때부터 다른 섬을 찾는 bfs2를 돌린다.

3. bfs2는 한 턴에 큐의 사이즈 만큼 돌게끔 구현하며 0의 위치를 계속해서 큐에 담는다.

4. map에서 0이 아니고 다른 섬 번호를 만나면 bfs2를 끝내고 min_cnt를 갱신한다.

5. 최솟값의 경우 시간을 줄여주기 위해서 중간에 cnt가 현재 min_cnt를 초과하면 더 이상 볼필요 없기 때문에 bfs2를 끝내준다.


소스코드

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
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;
 
public class Main {
 
    static class pair {
        int r;
        int c;
 
        public pair(int r, int c) {
            super();
            this.r = r;
            this.c = c;
        }
    }
 
    private static boolean[][] visited;
    private static int[][] map;
    private static int[] dr = { -1010 };
    private static int[] dc = { 010-1 };
    private static int N;
    private static int cnt;
    private static int min_cnt;
 
    public static void main(String[] args) throws Exception {
        // 입력
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        N = Integer.parseInt(br.readLine().trim());
        min_cnt = Integer.MAX_VALUE;
        map = new int[N][N];
        visited = new boolean[N][N];
 
        for (int i = 0; i < N; i++) {
            StringTokenizer st = new StringTokenizer(br.readLine().trim(), " ");
            for (int j = 0; j < N; j++) {
                map[i][j] = Integer.parseInt(st.nextToken());
            }
        }
        // 섬 별로 분류 bfs 이용
        int number = 1;
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < N; j++) {
                if (map[i][j] != 0 && !visited[i][j]) {
                    bfs1(i, j, number);
                    number++;
                }
            }
        }
 
        // 섬에서 0과 인접한 곳을 발견하면 그때부터는 0을 따라 가면서 다른 섬을 발견할 때 까지 카운트 해준다
        visited = new boolean[N][N];
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < N; j++) {
                if (map[i][j] != 0 && !visited[i][j]) {
                    visited[i][j] = true;
                    // 인접한 곳 중에서 0이 한 개라도 있으면
                    for (int k = 0; k < 4; k++) {
                        int nr = i + dr[k];
                        int nc = j + dc[k];
                        if (nr >= 0 && nr < N && nc >= 0 && nc < N) {
                            if (map[nr][nc] == 0) {
                                bfs2(map[i][j], nr, nc);
                                min_cnt = min_cnt < cnt ? min_cnt : cnt;
                            }
                        }
                    }
                }
            }
        }
 
        // 가장 짧게 카운트된 값을 구한다
        System.out.println(min_cnt);
 
    } // end of main
 
    private static void bfs2(int f_number, int r, int c) {
        Queue<pair> q = new LinkedList<>();
        boolean[][] zero_visited = new boolean[N][N];
        q.add(new pair(r, c));
        zero_visited[r][c] = true;
        cnt = 0;
 
        while (!q.isEmpty()) {
            int q_size = q.size();
            cnt++;
            if (cnt >= min_cnt)
                return;
            for (int size = 0; size < q_size; size++) {
                pair p = q.poll();
                for (int i = 0; i < 4; i++) {
                    int nr = p.r + dr[i];
                    int nc = p.c + dc[i];
                    if (nr >= 0 && nr < N && nc >= 0 && nc < N) {
                        if (map[nr][nc] == 0 && !zero_visited[nr][nc]) {
                            q.add(new pair(nr, nc));
                            zero_visited[nr][nc] = true;
                        } else if (map[nr][nc] != 0 && map[nr][nc] != f_number) {
                            // 다른섬을 만난경우
                            return;
                        }
                    }
                }
            }
        }
 
    } // end of bfs2
 
    private static void bfs1(int r, int c, int number) {
        Queue<pair> q = new LinkedList<>();
        q.add(new pair(r, c));
        visited[r][c] = true;
        map[r][c] = number;
 
        while (!q.isEmpty()) {
            pair p = q.poll();
            for (int i = 0; i < 4; i++) {
                int nr = p.r + dr[i];
                int nc = p.c + dc[i];
                if (nr >= 0 && nr < N && nc >= 0 && nc < N) {
                    if (map[nr][nc] != 0 && !visited[nr][nc]) {
                        q.add(new pair(nr, nc));
                        visited[nr][nc] = true;
                        map[nr][nc] = number;
                    }
                }
            }
        }
    } // end of bfs1
// end of class
cs

'APS' 카테고리의 다른 글

BOJ16234_인구 이동  (0) 2019.06.14
BOJ17144_미세먼지 안녕!  (0) 2019.06.10
BOJ17281_야구  (0) 2019.06.09
BOJ14891_톱니바퀴  (2) 2019.06.04
BOJ14499_주사위굴리기  (0) 2019.06.03