반응형
11-22 12:55
Today
Total
«   2024/11   »
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
관리 메뉴

개발하는 고라니

[백준] 16932번 : 모양 만들기 본문

Programming/백준

[백준] 16932번 : 모양 만들기

조용한고라니 2021. 4. 15. 17:32
반응형
 

16932번: 모양 만들기

N×M인 배열에서 모양을 찾으려고 한다. 배열의 각 칸에는 0과 1 중의 하나가 들어있다. 두 칸이 서로 변을 공유할때, 두 칸을 인접하다고 한다. 1이 들어 있는 인접한 칸끼리 연결했을 때, 각각의

www.acmicpc.net


[DFS]

백준의 다리 놓기(?) 시리즈 문제였나 그거랑 좀 비슷한 것 같다. 나의 문제 접근법은 다음과 같다.

 

  1. 지도를 입력 받으며 '0'인 곳은 Queue에 넣는다.
  2. DFS를 이용해 1로 이루어진 덩어리(모양)을 탐색하고, 그 덩어리에 ID(인덱스)를 부여하며, 그 id가 갖는 크기를 Map에 저장한다.
  3. Queue에서 하나씩 빼며 상하좌우 인접한 덩어리의 크기를 모두 더한다. (단, 덩어리가 중복되지 않게 하기 위해 Set을 이용했다.)
  4. 3번을 반복하며 max를 찾는다.

코드를 보면 조금 더 이해가 쉬울 것 같다.

# 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 r, c, max;
    static int[] Y = {-1,1,0,0}, X = {0,0,-1,1};
    static int[][] map = new int[1001][1001];
    static boolean[][] visit = new boolean[1001][1001];

    static Map<Integer, Integer> hashMap = new HashMap<>(); //각 덩어리의 id와 size를 갖음
    static Queue<Point> Q = new LinkedList<>();

    static int DFS(int y, int x, int id){

        visit[y][x] = true;
        map[y][x] = id;
        int result = 1;

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

            if(ny < 1 || nx < 1 || ny > r || nx > c || visit[ny][nx] || map[ny][nx] == 0) continue;
            result += DFS(ny, nx, id);
        }
        return result;
    }

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

        r = Integer.parseInt(st.nextToken());
        c = Integer.parseInt(st.nextToken());

        for(int i=1; i<=r; i++){
            st = new StringTokenizer(br.readLine());
            for(int j=1; j<=c; j++) {
                map[i][j] = Integer.parseInt(st.nextToken());
                if(map[i][j] == 0)
                    Q.add(new Point(i, j));
            }
        }
        /* DFS를 통한 각 덩어리(모양)의 인덱스를 부여하고, 그 크기를 구한다. */
        int id = 1;
        for(int i=1; i<=r; i++)
            for(int j=1; j<=c; j++)
                if(map[i][j] != 0 && !visit[i][j]) {
                    int size = DFS(i, j, id);
                    hashMap.put(id++, size);
                }
        /* Queue에 넣은 0을 하나씩 꺼내 1로 변경했을 때 인접한 덩어리들과 크기를 합쳐본다. */
        Set<Integer> set = new HashSet<>();//set을 이용해 동일한 덩어리의 중복 방지
        while(!Q.isEmpty()){
            Point p = Q.poll();

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

                if(ny < 1 || nx < 1 || ny > r || nx > c || map[ny][nx] == 0) continue;

                if(set.add(map[ny][nx]))
                    value += hashMap.get(map[ny][nx]);
            }
            max = Math.max(max, value);
            set.clear();
        }
        System.out.println(max);
    }
}
반응형

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

[백준] 1938번 : 통나무 옮기기  (0) 2021.04.30
[백준] 11021번 : A + B - 7  (0) 2021.04.28
[백준] 16973번 : 직사각형 탈출  (0) 2021.04.15
[백준] 14923번 : 미로 탈출  (0) 2021.04.15
[백준] 18119번 : 단어 암기  (0) 2021.04.09
Comments