자바칩
[백준 / Java] 1926: 그림 (BFS) 본문
728x90
난이도: Silver 1
문제: https://www.acmicpc.net/problem/1926
문제
어떤 큰 도화지에 그림이 그려져 있을 때, 그 그림의 개수와, 그 그림 중 넓이가 가장 넓은 것의 넓이를 출력하여라. 단, 그림이라는 것은 1로 연결된 것을 한 그림이라고 정의하자. 가로나 세로로 연결된 것은 연결이 된 것이고 대각선으로 연결이 된 것은 떨어진 그림이다. 그림의 넓이란 그림에 포함된 1의 개수이다.
입력
첫째 줄에 도화지의 세로 크기 n(1 ≤ n ≤ 500)과 가로 크기 m(1 ≤ m ≤ 500)이 차례로 주어진다. 두 번째 줄부터 n+1 줄 까지 그림의 정보가 주어진다. (단 그림의 정보는 0과 1이 공백을 두고 주어지며, 0은 색칠이 안된 부분, 1은 색칠이 된 부분을 의미한다)
출력
첫째 줄에는 그림의 개수, 둘째 줄에는 그 중 가장 넓은 그림의 넓이를 출력하여라. 단, 그림이 하나도 없는 경우에는 가장 넓은 그림의 넓이는 0이다.
BFS 알고리즘을 잘 이해했으면 쉽게 풀 수 있는 문제다.
넓이를 체크할 때는 넓이 체크 변수 area를 2로 초기화해서 새로운 칸에 방문할 때마다 area를 1씩 증가시키면 된다.
그리고 마지막에 방문한 x좌표와 y좌표의 넓이를 최댓값 변수에 갱신하면 된다.
주석에 한 줄 한줄 설명을 자세히 적어 놓았다.
코드를 바로 보자.
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
|
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;
public class P1926_그림 {
static int n, m; // 도화지의 세로, 가로 크기
static int[] dx = {-1, 0, 1, 0}; // 상, 하 이동
static int[] dy = {0, -1, 0, 1}; // 좌, 우 이동
static int[][] paper; // 2차원 도화지 배열
static boolean[][] visited; // 2차원 방문 배열
static int maxArea = 0; // 가장 넓은 그림의 넓이
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()); // 도화지의 가로 크기 입력 받기
paper = new int[n][m]; // 2차원 도화지 배열 크기 초기화
visited = new boolean[n][m]; // 2차원 방문 배열 크기 초기화
int count = 0; // 그림의 개수
// 2차원 도화지 배열 초기화
for (int i = 0; i < n; i++) {
st = new StringTokenizer(br.readLine());
for (int j = 0; j < m; j++) {
// 그림이 있는 경우: 1, 없는 경우: 0
paper[i][j] = Integer.parseInt(st.nextToken());
}
}
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
// 해당 칸을 방문하지 않았거나, 그림이 있는 칸이라면
if (!visited[i][j] && paper[i][j] != 0) {
count++; // 그림의 개수 1 증가
bfs(i, j); // 해당 칸부터 bfs 수행
}
}
}
System.out.println(count); // 그림의 개수 출력
System.out.println(maxArea); // 가장 넓은 그림의 넓이 출력
}
public static void bfs(int a, int b) {
Queue<int[]> queue = new LinkedList<>(); // bfs 수행을 위한 큐
queue.add(new int[] {a, b}); // 큐에 해당 칸 삽입
visited[a][b] = true; // 해당 칸을 방문 체크
int area = 2; // 해당 칸은 1이고, 다음 칸부터 넓이는 2로 카운트하면서 점점 증가
// 큐가 빌 때까지 반복
while (!queue.isEmpty()) {
int[] now = queue.poll(); // 현재 칸을 큐에서 뽑기
for (int i = 0; i < 4; i++) {
int x = now[0] + dx[i]; // 인접 칸의 x좌표
int y = now[1] + dy[i]; // 인접 칸의 y좌표
// 인접 칸의 x좌표나 y좌표가 유효한 인덱스가 아니라면 건너뛰기
if (x < 0 || y < 0 || x >= n || y >= m) {
continue;
}
// 인접 칸을 방문했거나, 그림이 있는 칸이 아니라면 건너뛰기
if (visited[x][y] || paper[x][y] == 0) {
continue;
}
queue.add(new int[] {x, y}); // 큐에 인접 칸 삽입
visited[x][y] = true; // 인접 칸을 방문 체크
paper[x][y] = area++; // 인접 칸을 area로 바꾸고 1 증가 => 넓이 체크
a = x; // 가장 최근 방문한 x좌표를 인자에 대입
b = y; // 가장 최근 방문한 y좌표를 인자에 대입
}
}
// 가장 넓은 그림의 넓이 갱신
// a: 가장 최근 방문한 x좌표, b: 가장 최근 방문한 y좌표
maxArea = Math.max(maxArea, paper[a][b]);
}
}
|
cs |
'알고리즘 > 백준' 카테고리의 다른 글
[백준 / Java] 14940: 쉬운 최단거리 (BFS) (0) | 2024.06.18 |
---|---|
[백준 / Java] 1743: 음식물 피하기 (BFS) (0) | 2024.06.17 |
[백준 / Java] 2583: 영역 구하기 (BFS) (1) | 2024.06.13 |
[백준 / Java] 13913: 숨바꼭질 4 (BFS) (0) | 2024.06.09 |
[백준 / Java] 14002: 가장 긴 증가하는 부분 수열 4 (DP) (0) | 2024.06.08 |