알고리즘 문제
[백준] 2636 치즈
feelcoding
2020. 8. 29. 22:21
728x90
https://www.acmicpc.net/problem/2636
2636번: 치즈
첫째 줄에는 사각형 모양 판의 세로와 가로의 길이가 양의 정수로 주어진다. 세로와 가로의 길이는 최대 100이다. 판의 각 가로줄의 모양이 윗 줄부터 차례로 둘째 줄부터 마지막 줄까지 주어진��
www.acmicpc.net
#include <vector>
#include <algorithm>
#include <iostream>
#include <stack>
using namespace std;
int main() {
cin.tie(NULL);
ios_base::sync_with_stdio(false);
int n, m;
int cnt = 0;
vector<int> countVector;
cin >> n >> m;
int temp;
vector<vector<int>> v(n, vector<int>(m));
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
cin >> temp;
v[i][j] = temp;
if (v[i][j] == 1) {
cnt++;
}
}
}
countVector.push_back(cnt);
int hour = 0;
vector<vector<bool>> visited(n, vector<bool>(m));
stack<pair<int, int>> s;
s.push(make_pair(0, 0));
while (!s.empty()) {
pair<int, int> cur = s.top();
s.pop();
int row = cur.first;
int col = cur.second;
if (!visited[row][col]) {
visited[row][col] = true;
v[row][col] = -1;
if (row + 1 < n && v[row + 1][col] == 0) {
s.push(make_pair(row + 1, col));
}
if (row - 1 >= 0 && v[row - 1][col] == 0) {
s.push(make_pair(row - 1, col));
}
if (col + 1 < m && v[row][col + 1] == 0) {
s.push(make_pair(row, col + 1));
}
if (col - 1 >= 0 && v[row][col - 1] == 0) {
s.push(make_pair(row, col - 1));
}
}
}
while (true) {
hour++;
for (int i = 1; i < n - 1; i++) {
for (int j = 1; j < m - 1; j++) {
if (v[i][j] == 1 && (v[i + 1][j] == -1 || v[i - 1][j] == -1 || v[i][j + 1] == -1 || v[i][j - 1] == -1)) {
v[i][j] = 2;
}
}
}
for (int i = 1; i < n - 1; i++) {
for (int j = 1; j < m - 1; j++) {
if (v[i][j] == 2) {
v[i][j] = -1;
}
}
}
cnt = 0;
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
if (v[i][j] == 1) {
cnt++;
}
}
}
if (cnt == 0)
break;
countVector.push_back(cnt);
visited = vector<vector<bool>>(n, vector<bool>(m));
s.push(make_pair(0, 0));
while (!s.empty()) {
pair<int, int> cur = s.top();
s.pop();
int row = cur.first;
int col = cur.second;
if (!visited[row][col]) {
visited[row][col] = true;
v[row][col] = -1;
if (row + 1 < n && v[row + 1][col] != 1) {
s.push(make_pair(row + 1, col));
}
if (row - 1 >= 0 && v[row - 1][col] != 1) {
s.push(make_pair(row - 1, col));
}
if (col + 1 < m && v[row][col + 1] != 1) {
s.push(make_pair(row, col + 1));
}
if (col - 1 >= 0 && v[row][col - 1] != 1) {
s.push(make_pair(row, col - 1));
}
}
}
}
cout << hour << '\n';
cout << countVector[countVector.size() - 1];
return 0;
}
DFS로 풀었다. 치즈 있는 부분은 1, 아무것도 없는 부분은 -1, 치즈의 구멍은 0, 녹아 없어질 치즈의 테두리 부분은 2로 표시했다. 주의해야 할 것은 1시간만에 치즈가 녹아 없어질 경우 초기에 주어진 치즈의 면적을 출력해야 하므로 미리 구해놓아야 한다는 것이다.
728x90