알고리즘
파이선) 파이프 옮기기 - BOJ
1일1공부실천하자
2023. 12. 13. 22:42
https://www.acmicpc.net/problem/17070
이 문제는 총 세 번의 재귀함수로 풀어봤었다.
그러다 모두 80퍼센트 쯤에서 시간초과가 발생했다.
첫번째 코드
import sys
input = sys.stdin.readline
n = int(input())
arr = []
for _ in range(n):
arr.append(list(map(int, input().split())))
answer = 0
def dfs(x, y, dir):
if x == n-1 and y == n-1:
global answer
answer += 1
return
if dir == 'row':
if y + 1 < n and arr[x][y+1] == 0:
dfs(x, y+1, 'row')
if x + 1 < n and y + 1 < n and arr[x+1][y+1] == 0 and arr[x+1][y] == 0 and arr[x][y+1] == 0:
dfs(x+1, y+1, 'dia')
elif dir == 'dia':
if y + 1 < n and arr[x][y+1] == 0:
dfs(x, y + 1, 'row')
if x + 1 < n and y + 1 < n and arr[x+1][y+1] == 0 and arr[x+1][y] == 0 and arr[x][y+1] == 0:
dfs(x+1, y + 1, 'dia')
if x + 1 < n and arr[x+1][y] == 0:
dfs(x+1, y, 'col')
else:
if x + 1 < n and arr[x+1][y] == 0:
dfs(x+1, y, 'col')
if x + 1 < n and y + 1 < n and arr[x+1][y+1] == 0 and arr[x+1][y] == 0 and arr[x][y+1] == 0:
dfs(x+1, y+1, 'dia')
dfs(0, 1, 'row')
print(answer)
두번째 코드
input = sys.stdin.readline
n = int(input())
arr = []
for _ in range(n):
arr.append(list(map(int, input().split())))
answer = 0
def dfs(x, y, dir):
if x == n-1 and y == n-1:
global answer
answer += 1
return
if x+1 < n and y + 1 < n:
if arr[x+1][y+1] == 0 and arr[x][y+1] == 0 and arr[x+1][y] == 0:
dfs(x+1, y+1, 2)
if dir == 0 or dir == 2:
if y + 1 < n:
if arr[x][y+1] == 0:
dfs(x, y+1, 0)
if dir == 1 or dir == 2:
if x + 1 < n:
if arr[x+1][y] == 0:
dfs(x+1, y, 1)
dfs(0, 1, 0)
print(answer)
그리고 마지막 세번째 코드...
import sys
input = sys.stdin.readline
n = int(input())
board = [list(map(int, input().split())) for _ in range(n)]
HO = 0 #가로
VIR = 1 #세로
DIAG = 2 #대각
cnt = 0
def dfs(nowy, nowx, type):
global cnt
if (nowy, nowx) == (n-1, n-1):
cnt += 1
return
if type == HO or type == DIAG:
if nowx+1 < n and board[nowy][nowx+1] == 0:
dfs(nowy, nowx+1, HO)
if type == VIR or type == DIAG:
if nowy + 1 < n and board[nowy+1][nowx] == 0:
dfs(nowy+1, nowx, VIR)
if nowx+1<n and nowy+1<n:
if board[nowy+1][nowx] == 0 and board[nowy][nowx+1] == 0 and board[nowy+1][nowx+1] == 0:
dfs(nowy+1, nowx+1, DIAG)
dfs(0, 1, HO)
print(cnt)
죄다 시간초과가 발생했다...
정답 코드 중 DP가 있어서 나 또한 DP를 이용해 풀었다.
def solution():
# 1행 미리 처리하기 → (3) 과정
dp[0][0][1] = 1
for i in range(2, N):
if board[0][i] == 0:
dp[0][0][i] = dp[0][0][i - 1]
# 왜 1행과 1열을 제외하는지는 (3), (4) 과정에서 봤었죠?
for r in range(1, N):
for c in range(1, N):
# (5) 과정
# 대각선 파이프를 추가하는 과정
if board[r][c] == 0 and board[r][c - 1] == 0 and board[r - 1][c] == 0:
dp[1][r][c] = dp[0][r - 1][c - 1] + dp[1][r - 1][c - 1] + dp[2][r - 1][c - 1]
# 가로, 세로 파이프를 추가하는 과정
if board[r][c] == 0:
dp[0][r][c] = dp[0][r][c - 1] + dp[1][r][c - 1]
dp[2][r][c] = dp[2][r - 1][c] + dp[1][r - 1][c]
# 최종 결과 출력
print(sum(dp[i][N - 1][N - 1] for i in range(3)))
N = int(input())
board = [list(map(int, input().split())) for _ in range(N)]
dp = [[[0 for _ in range(N)] for _ in range(N)] for _ in range(3)]
solution()
참... 이게 어떠게 골드 5 문제지...?