JUINTINATION
백준 7576번: 토마토 본문
문제
https://www.acmicpc.net/problem/7576
풀이
익은 토마토는 앞, 뒤, 왼쪽, 오른쪽 네 방향에 인접해있는 익지 않은 토마토를 하루가 지나면 익게 만들 수 있을 때 모두 익을 때까지 최소 날짜를 구하는 문제입니다.
코드
자바
import java.io.IOException;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Queue;
import java.util.LinkedList;
import java.util.StringTokenizer;
class point {
int x, y;
public point(int x, int y) {
this.x = x;
this.y = y;
}
}
public class Main {
static int n, m, max = 1;
static int[][] arr;
static ArrayList<point> list;
public static void bfs() { // (2)
Queue<point> queue = new LinkedList<>();
for (point p : list) {
queue.offer(p);
}
int[] dx = { -1, 0, 1, 0 };
int[] dy = { 0, -1, 0, 1 };
while (!queue.isEmpty()) {
point p = queue.poll();
max = Math.max(max, arr[p.x][p.y]); // (3)
for (int i = 0; i < 4; i++) {
int nx = p.x + dx[i];
int ny = p.y + dy[i];
if (0 <= nx && nx < m && 0 <= ny && ny < n && arr[nx][ny] == 0) { // (4)
arr[nx][ny] = arr[p.x][p.y] + 1;
queue.offer(new point(nx, ny));
}
}
}
}
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());
arr = new int[m][n];
list = new ArrayList<>();
for (int i = 0; i < m; i++) {
st = new StringTokenizer(br.readLine(), " ");
for (int j = 0; j < n; j++) {
arr[i][j] = Integer.parseInt(st.nextToken());
if (arr[i][j] == 1) list.add(new point(i, j)); // (1)
}
}
bfs();
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
if (arr[i][j] == 0) { // (5)
System.out.println(-1);
System.exit(0);
}
}
}
System.out.println(max - 1);
}
}
2차원 배열 arr는 처음에는 해당 위치의 토마토가 익었는지 여부를 나타냅니다.
(1) Arraylist list에는 최초에 익은 토마토가 있는 위치가 들어있습니다.
입력이 들어올 때 해당 위치의 토마토가 익은 상태라면 list에 추가합니다. 이후 bfs 함수로 들어갑니다.
bfs 함수 안에서 Queue는 익은 토마토의 위치 값(x, y)이 들어갑니다.
(2) bfs 함수 안에서 큐에 list의 모든 위치 값을 넣어줍니다.
이후 while문 안에서 큐가 비어있지 않을 때까지 (3)부터 (4) 과정을 반복합니다.
(3) max 값과 현재 탐색 중인 토마토가 익을 때까지 걸린 날짜를 비교해서 어떤 토마토가 익을 때까지 걸린 날짜의 최댓값을 구합니다.
이때 bfs 함수 내부에서의 arr는 해당 위치의 토마토가 익을 때까지 걸린 날짜 + 1을 의미하기 때문에 큐에서 poll한 위치 값을 p라고 할 때 arr[p.x][p.y]와 비교합니다.
(4) 그다음 탐색 위치(nx, ny)가 가능한 위치이며 해당 위치의 토마토가 익었는지 여부를 확인합니다.
익지 않았다면 현재 토마토에 의하여 다음날 익을 예정이므로 arr[nx][ny]에 arr[p.x][p.y] + 1을 넣어준 뒤에 (nx, ny)를 큐에 삽입합니다.
(5) bfs 함수가 종료되면 익지 않은 토마토가 있는지 확인합니다.
익지 않은 토마토가 1개라도 있으면 문제 조건에 맞지 않으므로 -1을 출력한 후에 프로그램을 종료하고 1개도 없다면 max 값에 1을 뺀 값을 출력하고 프로그램을 종료합니다.
C언어
#include <stdio.h>
#include <stdlib.h>
#define math_max(a, b) a > b ? a : b
int n, m, max = 1, **arr;
typedef struct {
int x, y;
} Point;
typedef struct node {
Point point;
struct node* next;
} node;
typedef struct Queue {
node* front;
node* rear;
int count;
} Queue;
void initQueue(Queue* queue) {
queue->front = queue->rear = NULL;
queue->count = 0;
}
void push(Queue* queue, int x, int y) {
node* now = (node*)malloc(sizeof(node));
now->point.x = x;
now->point.y = y;
now->next = NULL;
if (queue->count == 0) {
queue->front = now;
}
else {
queue->rear->next = now;
}
queue->rear = now;
queue->count++;
}
Point pop(Queue* queue) {
Point re;
node* now;
now = queue->front;
re = now->point;
queue->front = now->next;
free(now);
queue->count--;
return re;
}
int empty(Queue* queue) {
if (queue->count == 0) return 1;
else return 0;
}
void bfs() {
Queue queue;
initQueue(&queue);
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
if (arr[i][j] == 1) {
push(&queue, i, j);
}
}
}
int dx[] = { -1, 0, 1, 0 };
int dy[] = { 0, -1, 0, 1 };
while (!empty(&queue)) {
Point p = pop(&queue);
max = math_max(max, arr[p.x][p.y]);
for (int i = 0; i < 4; i++) {
int nx = p.x + dx[i];
int ny = p.y + dy[i];
if (0 <= nx && nx < m && 0 <= ny && ny < n && arr[nx][ny] == 0) {
arr[nx][ny] = arr[p.x][p.y] + 1;
push(&queue, nx, ny);
}
}
}
}
main() {
scanf("%d %d", &n, &m);
arr = (int**)malloc(sizeof(int*) * m);
for (int i = 0; i < m; i++) {
arr[i] = (int*)malloc(sizeof(int) * n);
for (int j = 0; j < n; j++) {
scanf("%d", &arr[i][j]);
}
}
bfs();
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
if (arr[i][j] == 0) {
printf("-1");
exit(0);
}
}
}
printf("%d", max - 1);
}
자바를 이용한 풀이와 동일합니다. list는 따로 구현하지 않고 bfs 함수 내부에서 2중 for문을 활용하여 최초에 익은 상태의 토마토의 위치를 큐에 삽입했습니다.
결론
2차원 배열을 그래프로 표현하여 시작점이 여러 개인 bfs 문제를 풀어봤습니다. 처음에 이러한 문제를 봤을 때 2차원 배열을 어떻게 1칸씩 이동하며 탐색할지 몰라서 당황했지만 비슷한 문제를 계속 풀다 보니 충분히 익숙해진 것 같습니다.
'백준 알고리즘 > 그래프와 순회' 카테고리의 다른 글
백준 2206번: 벽 부수고 이동하기 (0) | 2022.08.16 |
---|---|
백준 16928번: 뱀과 사다리 게임 (0) | 2022.08.12 |
백준 7569번: 토마토 (0) | 2022.08.10 |
백준 13023번: ABCDE (0) | 2022.08.07 |
백준 1260번: DFS와 BFS (0) | 2022.08.06 |