[DFS] 양팔저울

YangJiWon·2020년 11월 4일
0

알고리듬

목록 보기
5/8

양팔저울(DFS)

무게가 서로 다른 K개의 추와 빈 그릇이 있다. 모든 추의 무게는 정수이고, 그릇의 무게는 0
으로 간주한다. 양팔저울을 한 번만 이용하여 원하는 물의 무게를 그릇에 담고자 한다.
주어진 모든 추 무게의 합을 S라 하자. 예를 들어, 추가 3개이고, 각 추의 무게가 {1, 2, 6}이
면, S=9이고, 양팔저울을 한 번만 이용하여 1부터 S사이에 대응되는 모든 무게의 물을 다음과
같이 그릇에 담을 수 있다. X는 그릇에 담는 물의 무게이고, ⎕은 그릇을 나타낸다.

만약 추의 무게가 {1, 5, 7}이면 S=13이고, 그릇에 담을 수 있는 물의 무게는 {1, 2, 3, 4, 5,
6, 7, 8, 11, 12, 13}이고, 1부터 S사이에서 무게에서 9와 10에 대응하는 무게의 물을 담을
수 없다.
K(3<=K<=13)개의 추 무게가 주어지면, 1부터 S사이의 정수 중 측정이 불가능한 물의 무게는
몇 가지가 있는 지 출력하는 프로그램을 작성하세요.

▣ 입력설명

첫 번째 줄에 자연수 K(3<=K<=13)이 주어집니다.
두 번째 줄에 K개의 각 추의 무게가 공백을 사이에 두고 주어집니다. 각 추의 무게는 1부터
200,000까지이다.

▣ 출력설명

첫 번째 측정이 불가능한 가지수를 출력하세요.

▣ 입력예제 1

3
1 5 7

▣ 출력예제 1

2

풀이

def dfs(a, b):
    if a == k:
        if 1 <= b <= s and check[b] == 0:
            lst.append(b)
            check[b] = 1
    else:
        dfs(a + 1, b + w[a])
        dfs(a + 1, b - w[a])
        dfs(a + 1, b)


k = int(input())
w = list(map(int, input().split()))
s = sum(w)
lst = []
check = [0] * (s + 1)
dfs(0, 0)
print(s - len(lst))
  1. 무게를 더하는 경우
  2. 무게를 빼는 경우
  3. 무게를 더하지도 빼지도 않는 경우

이 세 가지로 나누어서 코드를 작성하였습니다. 그리고 lst에 하나씩 append하는 방식으로 작성을 하였고, append한 값은 check 리스트에 체크해가면서 다시 넣을 수 없도록 만들었습니다.

하지만 이 방법 보다 더 좋은 방법이 있습니다.

def dfs(a, b):
    if a == k:
        if 1 <= b <= s:
            nums.add(b)
    else:
        dfs(a + 1, b + w[a])
        dfs(a + 1, b - w[a])
        dfs(a + 1, b)


k = int(input())
w = list(map(int, input().split()))
s = sum(w)
nums = set()
check = [0] * (s + 1)
dfs(0, 0)
print(s - len(nums))

이 경우는 리스트로 하나씩 추가하는 경우 check리스트를 만들지 않고도 할 수 있는 방법입니다.

바로 set을 이용한 방법인데, check를 만들지 않더라도 set은 중복된 값을 만들지 않으므로 무조건 넣어주는 방식입니다.

오히려 앞에 방법보다 이 방식이 좋은 것 같습니다.

profile
데이터데이터데이터!!

0개의 댓글