[백준] 17472 : 다리 만들기 2 - Python

Chooooo·2024년 3월 30일
0

알고리즘/백준

목록 보기
152/204

문제 다리 만들기

섬 좌표 주어지고, 해당 섬들을 이을 수 있는 다리가 무한대로 존재한다.
해당 섬 좌표를 잇는 다리 길이의 최솟값 구하기

바다와 땅으로 이루어진 격자에서 모든 섬을 다리로 연결할 수 있을 때, 다리 길이의 최솟값을 구해보자.

1-1. 다리는 바다에만 건설가능
1-2. 다리의 방향에 중간에 바뀌면 안된다. (직선만 가능)
1-3. 다리의 길이는 2이상이어야 한다.

인접해 있는 땅은 하나로 묶는다 -> Flood-Fill (연결요소)
각 땅의 좌표들을 탐색하며 섬 번호 부여

내 코드

import sys
from collections import deque, defaultdict

sys.stdin = open("input.txt", "rt")

n,m = map(int, input().split())
g = [list(map(int, input().split())) for _ in range(n)]

dx = [-1,0,1,0]
dy = [0,1,0,-1]

island = defaultdict(int) # 각 좌표별 섬 번호 기록을 위해 딕셔너리(맵) 사용

def isInner(x,y):
    if 0<=x<n and 0<=y<m:
        return True
    return False

def BFS(a,b,num): # 시작 좌표, 섬번호 -> 해당 연결요소들 모두 num으로
    global visited
    dq = deque()
    dq.append((a,b))
    visited[a][b] = True
    island[(a,b)] = num
    while dq:
        x,y = dq.popleft()
        for i in range(4):
            nx = x + dx[i]
            ny = y + dy[i]
            if isInner(nx,ny) and g[nx][ny] == 1 and not visited[nx][ny]:
                visited[nx][ny] = True
                island[(nx,ny)] = num
                dq.append((nx,ny))

def get_min_dis(x,y): # 해당 좌표 기준 상하좌우 돌면서 확인
    src = island[(x,y)] # 현재 섬 번호
    # 재방문은 안된다.
    visited = [[False] * m for _ in range(n)]

    for i in range(4):
        nx,ny = x,y
        while True:
            nx += dx[i]
            ny += dy[i]
            if not isInner(nx,ny): break # 만약 좌표 벗어나면 탈출
            des = island[(nx,ny)]
            if des == src: break # 도착점과 같으면 안돼
            if g[nx][ny] == 1:
                d = abs(x-nx) + abs(y-ny) - 1
                if d >= 2 and d < dis[src][des]: # 다리 길이가 2이상이면서, 기존 값보다 작은 경우에 갱신
                    # print(f"시작 섬번호 {src} 도착 섬번호 {des} 거리 {d}")
                    dis[src][des] = d # 최단거리는 섬a, 섬b의 거리이므로 src->des 거리
                    dis[des][src] = d # 무방향 그래프이므로 역시 적용
                break # 사실 처음 1을 만난 순간 그 뒤는 확인하면 안된다.


def step2(): # 섬별 최단거리 구하기
    global dis
    for i in range(1, num + 1):
        dis[i][i] = 0
    for i in range(n):
        for j in range(m):
            if g[i][j] == 1:
                get_min_dis(i,j)

# 1. 섬 좌표별 몇번 섬인지 기록
num = 0 # 섬 개수 및 번호
visited = [[False] * m for _ in range(n)]
for i in range(n):
    for j in range(m):
        if g[i][j] == 1 and not visited[i][j]:
            num += 1
            BFS(i,j,num) # ! 1. step1 : 각 좌표 별 섬 번호 구하기

# 2. 좌표별 섬 번호 구했으니 섬 별 최단거리 구하기
INF = int(1e9)
dis = [[INF] * (num+1) for _ in range(num+1)] # 섬 개수만큼 만들어야한다.
step2() # step2 : 섬별 최단거리 구하기

# 3. 각 섬별 최단거리 구했으니 이제 각 간선에 섬1,섬2,비용. 이런 식으로 적용 : 간선 리스트 만들기
edge = []
def step3():
    for i in range(1,num+1):
        for j in range(1,num+1):
            if i == j: continue
            if dis[i][j] != INF: # 값이 존재한다면
                edge.append((i,j,dis[i][j]))
    edge.sort(key = lambda x : x[2])

step3()

# 4. 크루스칼을 통해 계산하기
parent = [0] * (num+1)
for i in range(1,num+1):
    parent[i] = i
def find_parent(x):
    if x == parent[x]:
        return x
    parent[x] = find_parent(parent[x])
    return parent[x]
def union_parent(a,b):
    a = find_parent(a)
    b = find_parent(b)
    if a < b: parent[b] = a
    else: parent[a] = b

res = 0
for i in range(len(edge)):
    a,b,cost = edge[i]
    if find_parent(a) != find_parent(b):
        res += cost
        union_parent(a,b)

connected = True
root = find_parent(1)
for i in range(2,num+1):
    if root != find_parent(i):
        connected = False
        break

if connected:
    print(res)
else:
    print(-1)

코멘트

  1. 보드를 BFS를 돌려서 Flood-Fill(연결요소) 별로 섬 번호를 부여한다.
  • 이 과정에서 좌표를 key로 val을 섬 번호로 하는 dict를 사용했다.
  1. 각 섬끼리의 최단거리를 계산했다.
    2-1. 처음 만나는 1이 같은 섬번호가 아니면서
    2-2. 거리가 2 이상이어야 했다.
  • 이 과정에서 다른 섬의 1을 만나는 경우 계산 후 끝냈어야했다.. 왜냐하면 그 뒤 1은 확인하면 안됐거든... (1을 만난 후 로직 처리 후 바로 탈출)
  1. 이제 각 섬끼리의 가중치(cost, 비용)를 구했으니 edge에 가능한 경우를 모두 삽입 (a,b,cost) -> dis[a][b] != INF , a != b 인 경우
  2. 크루스칼 알고리즘 적용

최소신장트리에서 사이클을 만들지 않으면서 최단거리를 완성.

다시 풀어볼 문제 -> 다른 사람의 조언을 들어보니 굳이 순간순간 거리를 계산할 게 아니라 그냥 바로 edges에 넣어도 상관없었다고 한다. 왜냐하면 어차피 정렬 후 가장 짧은 애들부터 사용할거라 알아서 걸러질 것이라고 한다.

  • MST 다시 공부 예정

다시 풀어보자.

profile
back-end, 지속 성장 가능한 개발자를 향하여

0개의 댓글