Algorithm/DFS & BFS

(Python/파이썬) - 백준(BOJ) 1939번 : 중량제한

하눤석 2022. 4. 18. 10:11
728x90

https://www.acmicpc.net/problem/1939

 

1939번: 중량제한

첫째 줄에 N, M(1 ≤ M ≤ 100,000)이 주어진다. 다음 M개의 줄에는 다리에 대한 정보를 나타내는 세 정수 A, B(1 ≤ A, B ≤ N), C(1 ≤ C ≤ 1,000,000,000)가 주어진다. 이는 A번 섬과 B번 섬 사이에 중량제한이

www.acmicpc.net


  • 문제 : 

N(2 ≤ N ≤ 10,000)개의 섬으로 이루어진 나라가 있다. 이들 중 몇 개의 섬 사이에는 다리가 설치되어 있어서 차들이 다닐 수 있다.

 

영식 중공업에서는 두 개의 섬에 공장을 세워 두고 물품을 생산하는 일을 하고 있다. 물품을 생산하다 보면 공장에서 다른 공장으로 생산 중이던 물품을 수송해야 할 일이 생기곤 한다. 그런데 각각의 다리마다 중량제한이 있기 때문에 무턱대고 물품을 옮길 순 없다. 만약 중량제한을 초과하는 양의 물품이 다리를 지나게 되면 다리가 무너지게 된다.

 

한 번의 이동에서 옮길 수 있는 물품들의 중량의 최댓값을 구하는 프로그램을 작성하시오.

 

 

 


  • 풀이 :

이분탐색과 그래프 탐색 문제였습니다.

 

알고리즘의 흐름입니다.

 

1. 양방향 그래프 구현

 

2. 이분탐색의 범위 지정. 

  저의 경우에는 left = 1, right 는 입력받은 다리의 하중들의 max값을 사용하였습니다.

 

3. 이분탐색 구현. 

  이분탐색의 search값은 무게입니다. 택배의 무게를 mid로 잡고 이 값으로 bfs를 실시합니다.

 

4. bfs 구현

  bfs의 인자값은 weight로 이 값이 도착점으로 갈 수 있냐, 없냐를 따지는 방식으로 구현하였습니다.

bfs탐색이 종료된 후에 visited배열의 t (도착점)이 초깃값이라면 도착할 수 없으므로 False를, 아니라면 True를 리턴하게 하였습니다.

 

5. 이분탐색 범위 조정

 mid값에서 bfs결과 해당 무게는 도착점까지 옮길 수 있으면 left를 mid+1로 바꿔주어 범위를 재 조정하고 다시 bfs를 실시합니다. 이분탐색이 종료될 때까지 bfs를 실행하게 됩니다.

 


  • 소스코드 : 
import sys
from collections import deque
input = sys.stdin.readline
N,M = map(int,input().split())
graph = [[] for _ in range(N+1)]
left = 1
right = 0
answer = 0
def bfs(weight):  # 무게(wieght)를 기준으로 BFS탐색
    queue = deque()
    queue.append(f) #시작점은 무조건 f
    visited = [0 for _ in range(N+1)]
    visited[f] = 0
    while queue:
        x = queue.popleft()
        for i,w in graph[x]:
            if w >= weight and visited[i] == 0:
                queue.append(i)
                visited[i] = 1
    if visited[t] != 0: return True  #갈 수 있을 때
    return False # 못 갈 때

for _ in range(M): #양방향 그래프 구현
    a,b,c = map(int,input().split())
    right = max(right,c) # 이분 탐색의 끝 점을 정의하기 위함
    graph[a].append([b,c])
    graph[b].append([a,c])
f,t = map(int,input().split())  # 출발점, 도착점

while left <= right:
    mid = (left+right)//2
    if bfs(mid): # 출발점에서 도착점까지 무게 mid를 옮길 수 있는 경우
        answer = mid
        left = mid+1
    else: # 못 옮기는 경우
        right = mid -1
print(answer)
320x100