백준 2146번 다리 만들기 파이썬

박슬빈·2021년 11월 19일
0

알고리즘

목록 보기
27/40

문제


입력 , 출력

solution

import sys

sys.setrecursionlimit(10**5)
from collections import deque

input = sys.stdin.readline


def bfs(startx, starty):
    global var
    dq.append((startx, starty))
    visited[startx][starty] = var
    while dq:
        x, y = dq.popleft()
        for i in range(4):
            nx = x + dx[i]
            ny = y + dy[i]
            if 0 <= nx < n and 0 <= ny < n:
                if arr[nx][ny] == 1 and visited[nx][ny] == 0:
                    dq.append((nx, ny))
                    visited[nx][ny] = var
                if arr[nx][ny] == 0:
                    visited[x][y] = -var
    var += 1


def bfs2(m):
    dq2 = deque()
    dist = [[-1 for _ in range(n)] for i in range(n)]
    for i in range(n):
        for j in range(n):
            if -m == visited[i][j]:
                dq2.append((i, j))
                dist[i][j] = 0
    while dq2:
        x, y = dq2.popleft()
        for i in range(4):
            nx = x + dx[i]
            ny = y + dy[i]
            if 0 <= nx < n and 0 <= ny < n:
                if visited[nx][ny] != 0 and visited[nx][ny] != m and visited[
                        nx][ny] != -m:
                    return dist[x][y]
                if arr[nx][ny] == 0 and dist[nx][ny] == -1:
                    dq2.append((nx, ny))
                    dist[nx][ny] = dist[x][y] + 1


n = int(input())
dq = deque()
visited = [[0 for i in range(n)] for i in range(n)]
res = 100000
dx = [1, 0, -1, 0]
dy = [0, 1, 0, -1]
arr = [[0 for i in range(n)] for i in range(n)]
var = 2
for i in range(n):
    arr[i] = list(map(int, input().split()))
for i in range(n):
    for j in range(n):
        if visited[i][j] == 0 and arr[i][j] == 1:
            bfs(i, j)
for i in range(var - 2):
    res = min(res, bfs2(i + 2))
print(res)

설명

  1. bfs를 두번 사용
  2. 처음 bfs를 돌면서 섬의 번호를 붙임
  3. 외곽쪽에 섬의 번호를 -로 마킹을 함
  4. 섬의 개수만큼 bfs를 돌림
  5. 섬의 외곽 마킹한곳에서 부터 다른 섬이랑 이어질때까지 bfs를 돌림
  6. bfs를 돌렸을때 가장 적은 거리로 갱신을 해줌

bfs 할 때 주의
큐에서 빼면서 visited를 체킹하지말고
for문으로 좌우 양옆을 하면서 visited를 체크
WHY? 큐에 들어가 있는데 또 방문을 할수도 있어서 메모리초과가 날수도있음

profile
이것저것합니다

0개의 댓글