일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
- 2월공모주
- 백준 알고리즘
- 정렬
- 사이킷런
- 벽부수고이동하기 파이썬
- 사회조사분석사 2급 기출문제집
- 사회조사분석사 2급 독학
- 공모주
- 머신러닝
- 백준
- 현대엔지니어링 수요예측
- 현대엔지니어링
- DFS
- 너비우선탐색
- 사회조사분석사 2급 필기 시험시간
- 사회조사분석사 2급
- 사회조사분석사 2급 필기 공부방법
- 사회조사분석사2급실기신청꿀팁
- 오미크론 자가격리
- 그리디
- 공모주청약
- 사회조사분석사 2급 공부방법
- BFS
- 알고리즘
- 사회조사분석사2급실기신청
- 시물레이션
- 사회조사분석사 2급 필기 요약정리
- 공모주 청약
- 파이썬 정렬
- 사회조사분석사 2급 접수
- Today
- Total
세상을 바꾸는 데이터
[백준 16235번] 나무 재태크 - 파이썬 본문
문제 링크:
https://www.acmicpc.net/problem/16235
풀이 유형:
구현, 시물레이션
★문제풀이 Point★
1. 나무의 정보를 3차원 리스트로 담기 (행, 열, 나이)
2. 문제를 꼼꼼히 읽으면서 사계절별로 구현
3. 사계절 중에 여름 양분 처리 중요
풀이 과정:
이 문제는 K년이 지난 후 상도의 땅에 살아있는 나무의 개수를 구하는 프로그램을 작성하는 전형적인 시물레이션 문제이다.
구현, 시물레이션 문제는 문제를 꼼꼼히 읽어야 한다. 문장들이 다 코드에 녹아들기 때문이다.
1. 나무 정보를 받는 3차원 리스트(행, 열, 나무의 나이) tree에서 나무의 나이를 리스트가 아닌 deque으로 받는다.
이는 가을에 나무가 번식을 하여 나이가 1인 나무를 생성할 때 맨 앞부분에 넣기 위함(appendleft 사용)이다.
2. 문제에서 제한 부분을 살펴보면 다음과 같이 제한이 걸려있다.
다음 문장이 이 문제에 핵심 포인트이다.
이 문장은 굳이 나무의 나이를 기준으로 정렬할 필요가 없음을 알려준다.
즉, deque을 리스트로 바꿔서 정렬(sort)하고, 리스트를 다시 deque으로 바꿔야 할 필요가 없다.
3. 봄, 여름, 가을, 겨울 순으로 구현을 해주면 되는데, 여기서 필자가 가장 어려웠던 점은 여름이다.
# 여름
# 양분이 부족하다면
else:
# 뒤에 있는 모든 나무들이 양분이 부족하므로
for _ in range(k, tree_cnt):
# 전부 없애주고 양분으로 만들기
food[i][j] += tree[i][j].pop() // 2
break
각 위치에 있는 나무들이 오름차순으로 정렬 1)되어 있음으로, 양분이 부족할 때는 현재 나무부터 뒤에 있는 나무들을 전부 없애고 양분을 만들어주면 된다.
1) 오름차순 정렬 근거:
1. 위 2번에 해당하는 제한 문장으로 인해 처음 1년에는 입력값이 각 위치마다 나무가 최대 1개밖에 들어갈 수 없음.
2. 가을에 번식 후에는 나이가 1인 나무를 appendleft를 이용해 맨 앞으로 보내주므로 오름차순 정렬이 완성됨.
구현, 시물레이션 문제는 정말로 복잡한 문제이다. 하지만 이런 문제일수록 침착하게 풀어야 한다.
풀이 코드:
다음 코드는 PyPy3로 실행해 풀었다.
# 16235번 나무 재태크
from collections import deque
# 봄과 여름을 같이 처리하는 함수
def spring_summer():
# 각 위치마다 하나씩 살펴보며
for i in range(n):
for j in range(n):
# 현재 위치에 있는 트리 개수 세기
tree_cnt = len(tree[i][j])
# 트리 개수만큼 반복하며
for k in range(tree_cnt):
# 봄
# 현재 위치에 있는 양분이 나무 나이보다 많거나 같다면
if food[i][j] >= tree[i][j][k]:
# 양분 공급
food[i][j] -= tree[i][j][k]
# 나무 나이 1 증가
tree[i][j][k] += 1
# 여름
# 양분이 부족하다면
else:
# 뒤에 있는 모든 나무들이 양분이 부족하므로
for _ in range(k, tree_cnt):
# 전부 없애주고 양분으로 만들기
food[i][j] += tree[i][j].pop() // 2
break
return
# 가을과 겨울을 처리하는 함수
def fall_winter():
for i in range(n):
for j in range(n):
# 가을
# 현재 위치에 있는 나무의 나이를 차례대로 받아
for k in range(len(tree[i][j])):
# 나무의 나이가 5의 배수라면
if tree[i][j][k] % 5 == 0:
# 8가지 번식 이동방향을 확인
for l in range(8):
nx = i + dx[l]
ny = j + dy[l]
# 주어진 맵 안에서 번식이 가능하다면
if 0 <= nx < n and 0 <= ny < n:
# 나이가 1인 나무를 이동 위치에 우선적으로 추가
tree[nx][ny].appendleft(1)
# 겨울
# 양분 공급
food[i][j] += s2d2[i][j]
return
# 입력값 받기
n, m, k = map(int, input().split())
# 나무 정보를 받는 3차원 리스트 (행, 열, 나무의 나이)
tree = [[deque() for _ in range(n)] for _ in range(n)]
# 각 위치에 있는 양분 리스트
food = [[5] * n for _ in range(n)]
# 가을에 나무가 번식하는 8가지 방법
dx = [-1, -1, -1, 0, 0, 1, 1, 1]
dy = [-1, 0, 1, -1, 1, -1, 0, 1]
# 로봇 s2d2가 겨울에 추가하는 양분 리스트 입력받기
s2d2 = [list(map(int, input().split())) for _ in range(n)]
# 나무 정보 입력받기
for _ in range(m):
x, y, z = map(int, input().split())
# 나무 위치에 나무의 나이 정보 추가
tree[x-1][y-1].append(z)
# K년이 될 때까지 나무 재태크 실시
for _ in range(k):
spring_summer()
fall_winter()
# 나무의 개수 세기
answer = 0
for i in range(n):
for j in range(n):
answer += len(tree[i][j])
# 정답 출력
print(answer)
'PS Study > BOJ(백준)' 카테고리의 다른 글
[백준 17427번] 약수의 합 2 - 파이썬 (30) | 2022.03.07 |
---|---|
[백준 16954번] 움직이는 미로 탈출 - 파이썬 (31) | 2022.03.06 |
[백준 16946번] 벽 부수고 이동하기 4 - 파이썬 (56) | 2022.03.01 |
[백준 2206번] 벽 부수고 이동하기 - 파이썬 (40) | 2022.02.28 |
[백준 12886번] 돌 그룹 - 파이썬 (40) | 2022.02.26 |