반응형
01-11 16:15
Today
Total
«   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
관리 메뉴

개발하는 고라니

[백준] 16946번 : 벽 부수고 이동하기 4 본문

Programming/백준

[백준] 16946번 : 벽 부수고 이동하기 4

조용한고라니 2021. 3. 10. 03:27
반응형
 

16946번: 벽 부수고 이동하기 4

N×M의 행렬로 표현되는 맵이 있다. 맵에서 0은 이동할 수 있는 곳을 나타내고, 1은 이동할 수 없는 벽이 있는 곳을 나타낸다. 한 칸에서 다른 칸으로 이동하려면, 두 칸이 인접해야 한다. 두 칸이

www.acmicpc.net


[BFS]

Time Limit Exceeded(TLE)를 높은 확률로 만날 수 있는 문제이다. 여러 가지 방식으로 문제를 풀어봤으나 매번 시간 초과... 결국 AC를 받은 것은 다음과 같은 방법이다.

 

* 갈 수 있는 칸을 저장하는 int[][] move

* 빈 공간에 대해 방문 체크 boolean[][] visit

* 벽에 대해 방문 체크 boolean[][] wall

* 지도 저장 int[][] map

 

1. 입력을 받으며 벽인 곳의 move는 1로 세팅

2. 2중 루프를 돌며 빈 공간인 곳에 대해 BFS를 실시.

2-1. 갈 수 있는 빈 공간의 개수를 세고, 마주치는 벽의 정보를 List에 저장

3. BFS 종료 후 마주친 벽에 저장해둔 빈 공간의 개수를 더함

 

2 ~ 3 반복

 

처음에는 빈 공간의 개수를 만난 벽들에 더해주는 것이 아닌, 각각의 빈 공간이 그 주변의 빈 공간 개수를 저장하는 식으로 했다. 결과적으로 불필요한 반복을 더 하게 되어 시간초과를 만났다.

 

(+추가)

출력하는 부분에 있어서 System.out을 이용한 출력을 사용해왔는데, 이 문제처럼 출력할 수가 많아지면 StringBuilder를 이용하는 것이 시간단축에 있어 더 좋다는 글을 보고 추가로 작성한다. 실제로 System.out 을 사용해서 출력을 했을 떄와 확연하게 차이가 난다.

        for(int i=1; i<=n; i++){
            StringBuilder builder = new StringBuilder();
            
            for(int j=1; j<=m; j++)
                builder.append(move[i][j] % 10);
                
            System.out.println(builder.toString());
        }

# Code </>

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.*;

public class Main {

    static class Point {
        int y, x;
        public Point(int yy, int xx) {
            y = yy;
            x = xx;
        }
    }

    static int[][] map = new int[1001][1001], move = new int[1001][1001];
    static boolean[][] visit = new boolean[1001][1001], wall = new boolean[1001][1001];
    static int[] Y = {-1, 1, 0, 0}, X = {0, 0, -1, 1};
    static int n, m;

    static void BFS() {

        Queue<Point> Q = new LinkedList<>();
        List<Point> list = new ArrayList<>();

        for (int i = 1; i <= n; i++)
            for (int j = 1; j <= m; j++) {

                int cnt = 1;
                if (!visit[i][j] && map[i][j] == 0) {
                    list.clear();
                    visit[i][j] = true;
                    Q.add(new Point(i, j));

                    while (!Q.isEmpty()) { //빈 공간의 개수 구하기
                        Point p = Q.poll();
                        int y = p.y;
                        int x = p.x;

                        for (int a = 0; a < 4; a++) {
                            int ny = y + Y[a];
                            int nx = x + X[a];

                            if (ny < 1 || nx < 1 || ny > n || nx > m || visit[ny][nx]) continue;

                            if(map[ny][nx] == 1){
                                if(!wall[ny][nx]) {
                                    list.add(new Point(ny, nx)); //주변 벽의 좌표 저장
                                    wall[ny][nx] = true;
                                }
                                continue;
                            }

                            cnt++;
                            visit[ny][nx] = true;
                            Q.add(new Point(ny, nx));
                        }
                    }
                    for(Point p:list) { //저장한 벽에 빈 공간 개수 더해줌
                        move[p.y][p.x] += cnt;
                        wall[p.y][p.x] = false;
                    }
                }
            }
    }

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

        n = Integer.parseInt(st.nextToken());
        m = Integer.parseInt(st.nextToken());

        for(int i=1; i<=n; i++){
            char[] str = br.readLine().toCharArray();
            for(int j=1; j<=m ;j++) {
                map[i][j] = str[j - 1] - '0';

                if(map[i][j] > 0)
                    move[i][j] = 1;
            }
        }

        BFS();

        for(int i=1; i<=n; i++){
            for(int j=1; j<=m; j++)
                System.out.print(move[i][j] % 10);
            System.out.println();
        }
    }
}
반응형

'Programming > 백준' 카테고리의 다른 글

[백준] 10815번 : 숫자 카드  (0) 2021.03.12
[백준] 1963번 : 소수 경로  (0) 2021.03.12
[백준] 9328번 : 열쇠  (0) 2021.03.09
[백준] 1525번 : 퍼즐  (0) 2021.03.07
[백준] 14938번 : 서강그라운드  (0) 2021.03.06
Comments