CS/Algorithm
[BOJ][C++] 16235 나무재테크
별토끼.
2019. 2. 16. 15:01
반응형
[BOJ][C++] 16235 나무재테크
문제
https://www.acmicpc.net/problem/16235
풀이
시간초과 때문에 몇 번을 다시 풀었던 문제 입니다. 문제 풀면서 시간 고려도 하는 버릇을 길러야겠어요
2차원 배열에 벡터 형태로 나무를 쌓는 식(z축)으로 했습니다.
그리고 k년만큼 for문을 돌면서 봄, 여름, 가을, 겨울 을 순으로 코드를 짰어요.
봄 >
봄은 나이만큼 양분을 먹노 나이가 증가합니다. 나이만큼 양분이 없는 경우 여름에 비료로 쓸 수 있도록 dead배열을 하나 더 만들어 나이/2를 한 값을 저장해줍니다. 주의할 것은 나이가 어린 순 부터 비료를 쓸 수 있기에 sort를 해주었습니다.
여름 >
저장해 놓았던 dead배열의 값을 더해줍니다. 이 때, dead배열 초기화를 해줍니다.
가을 >
dx, dy 배열을 만들어 여덟 방향을 갈 수 있도록 해줍니다. 주의해야 할 것은 age배열에 나무를 추가해 심을 경우 나무의 수가 계속 변해서 오류가 날 수 있어요. 그렇기 때문에 임시 배열 하나를 만들어 복사해준 뒤 임시 배열에 나무를 심어줍니다. 5의 배수 나이를 갖고 있는 나무 주위로 모든 나무를 심어주고 나면 심은 나무를 다시 원본 배열에 복사해줍니다.
겨울>
p_compost배열에 저장해놨던 값을 compost배열에 더해줍니다.
k년 만큼 모두 돌면 age[i][j]에 있는 나무들의 size를 측정해서 다 더해주면 됩니다.
코드
#define _CRT_SECURE_NO_WARNINGS
#include <cstdio>
#include <vector>
#include <math.h>
#include <algorithm>
using namespace std;
int main() {
int n, m, k;
int p_compost[11][11];
vector<int> age[11][11];
int compost[11][11];
int dead_tree[11][11];
int dx[] = { 1,-1,0,0,1,-1,1,-1 };
int dy[] = { 0,0,1,-1,1,-1,-1,1 };
scanf("%d %d %d", &n, &m, &k);
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= n; j++) {
scanf("%d", &p_compost[i][j]);
compost[i][j] = 5;
}
}
//나무 입력받기
for (int i = 1; i <= m; i++) {
int x, y, z;
scanf("%d %d %d", &x, &y, &z);
age[x][y].push_back(z);
}
//년도 증가
for (int i = 0; i < k; i++) {
//1.봄
for (int p = 1; p <= n; p++) {
for (int q = 1; q <= n; q++) {
if (!age[p][q].empty()) {
sort(age[p][q].begin(), age[p][q].end());
for (int j = 0; j < age[p][q].size(); j++) {
if (compost[p][q] >= age[p][q][j]) {
compost[p][q] -= age[p][q][j];
age[p][q][j]++;
}
else {
dead_tree[p][q] += age[p][q][j] / 2;
age[p][q].erase(age[p][q].begin() + j);
j--;
}
}
}
}
}
//2.여름
for (int p = 1; p <= n; p++) {
for (int q = 1; q <= n; q++) {
compost[p][q] += dead_tree[p][q];
}
}
for (int p = 1; p <= n; p++) {
for (int q = 1; q <= n; q++) {
dead_tree[p][q] = 0;
}
}
//3.가을
vector<int> tmp[11][11];
for (int p = 1; p <= n; p++) {
for (int q = 1; q <= n; q++) {
for (int j = 0; j < age[p][q].size(); j++) {
tmp[p][q].push_back(age[p][q][j]);
}
}
}
for (int p = 1; p <= n; p++) {
for (int q = 1; q <= n; q++) {
for (int j = 0; j < age[p][q].size(); j++) {
if (age[p][q][j] % 5 == 0) {
for (int l = 0; l < 8; l++) {
int nx = p + dx[l];
int ny = q + dy[l];
if (nx <= 0 || ny <= 0 || nx > n || ny > n) continue;
tmp[nx][ny].push_back(1);
}
}
}
}
}
for (int p = 1; p <= n; p++) {
for (int q = 1; q <= n; q++) {
age[p][q].clear();
for (int j = 0; j < tmp[p][q].size(); j++) {
age[p][q].push_back(tmp[p][q][j]);
}
}
}
//4.겨울
for (int p = 1; p <= n; p++) {
for (int q = 1; q <= n; q++) {
compost[p][q] += p_compost[p][q];
}
}
}
int res = 0;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= n; j++) {
res += age[i][j].size();
}
}
printf("%d\n", res);
}
반응형