3 분 소요

📌 백준 2805 문제 링크

백준 2805 나무 자르기

높이 h 지정 -> 모두 잘라 -> 필요한 만큼만 집으로 가져가려고 하는데, 이때 적어도 M미터의 나무를 집에 가져가기 위해서 절단기에 설정할 수 있는 높이의 최댓값을 구하는 프로그램

실패 코드

# 2805 나무 자르기

# 높이 h 지정 -> 모두 잘라 -> 필요한 만큼만 집으로 가져가려고 하는데, 이때 적어도 M미터의 나무를 집에 가져가기 위해서 절단기에 설정할 수 있는 높이의 최댓값을 구하는 프로그램
import sys

input = sys.stdin.readline
n, m = map(int, input().split())
height = [int(i) for i in input().split()]

start = 1
end = max(height)
isPossible = False
while start <= end:
    mid = (start + end) // 2
    get = 0
    # 얻은 나무의 길이 구하기
    for i in range(n):
        # 나무가 자를 길이보다 짧다면 얻을 수 있는 길이가 없음.
        if height[i] - mid >= 0:
            get += height[i] - mid
    if get == m:
        print(mid)
        break
    if get < m:
        end = mid - 1
    else:
        start = mid + 1

정답(2%)

반례

3 4
3 3 5

필요한 길이랑 가장 비슷하게 잘라야되는 길이는 2이다…. 하지만 출력은 안되고 while 문 조건때문에 출력은 안되고 반복문이 끝남.

실패 코드

# 2805 나무 자르기

# 높이 h 지정 -> 모두 잘라 -> 필요한 만큼만 집으로 가져가려고 하는데, 이때 적어도 M미터의 나무를 집에 가져가기 위해서 절단기에 설정할 수 있는 높이의 최댓값을 구하는 프로그램
import sys

input = sys.stdin.readline
n, m = map(int, input().split())
height = [int(i) for i in input().split()]

start = 1
end = max(height)
mid = 0
while start <= end:
    mid = (start + end) // 2
    get = 0
    # 얻은 나무의 길이 구하기
    for i in range(n):
        # 나무가 자를 길이보다 짧다면 얻을 수 있는 길이가 없음.
        if height[i] - mid >= 0:
            get += height[i] - mid
    # print(get, m)
    if get == m:
        
        break
    if get < m:
        end = mid - 1
    else:
        start = mid + 1

print(mid)

무조건 나무를 가져가야되므로 반드시 잘린 나무가 M일 필요는 없음.

10 1
600000000 600000000 600000000 600000000 600000000 600000000 600000000 600000000 600000000 600000000

반례 발견

만약 저렇게 입력이 된다면 59999999 이 나와야 되는데, 나는 600000000 이 나온다.

실패 코드

# 2805 나무 자르기

# 높이 h 지정 - > 모두 잘라 - > 필요한 만큼만 집으로 가져가려고 하는데, 이때 적어도 M미터의 나무를 집에 가져가기 위해서 절단기에 설정할 수 있는 높이의 최댓값을 구하는 프로그램
import sys

input = sys.stdin.readline
n, m = map(int, input().split())
height = [int(i) for i in input().split()]

# 4 8
# 20 15 10 17

start = 1
end = max(height)
mid = 0
get = 0
while start < end:
    mid = (start + end) // 2
    # 얻은 나무의 길이 구하기# 나무가 자를 길이보다 짧다면 얻을 수 있는 길이가 없음.
    get = sum([(t - mid) if t - mid > 0 else 0 for t in height])

    # print(get, m, '자른 길이:', mid)
    if get == m:
        break
    if get < m:
        end = mid - 1
    else :
        start = mid + 1

print(mid)

여튼 자른게 나와야되긴 하니까 while 문 조건에 =을 붙히지는 않음. 근데 정답입니다(6%) 아니 뭐가 문제세여?????

정답 코드

아씨,, 댕 어이없네,,,,,, 마지막에 mid를 출력하는게 아닌 end를 출력,,,,,,,,,,,,,ㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋ

추가적으로 나무는 무조건 잘라야되므로 get 이 구해야되는 나무길이보다 길더라도 m과 가장 비슷한 길이를 구해야되므로 중간에 break 문 없이 계속 이분탐색을 돌리면 됨…

# 2805 나무 자르기

# 높이 h 지정 - > 모두 잘라 - > 필요한 만큼만 집으로 가져가려고 하는데, 이때 적어도 M미터의 나무를 집에 가져가기 위해서 절단기에 설정할 수 있는 높이의 최댓값을 구하는 프로그램
import sys

input = sys.stdin.readline
n, m = map(int, input().split())
height = [int(i) for i in input().split()]

# 4 8
# 20 15 10 17

start = 1
end = max(height)

while start <= end:
    mid = (start + end) // 2
    # 얻은 나무의 길이 구하기
    # 나무가 자를 길이보다 짧다면 얻을 수 있는 길이가 없음.
    get = sum([(t - mid) if t - mid > 0 else 0 for t in height])

    # print(get, m, '자른 길이:', mid)
    if get >= m:
        start = mid + 1
        
    elif get < m:
        end = mid - 1

print(end)

코드 설명

일반적인 이분 탐색 문제이다. 당연히 자르려고 하는 나무의 길이를 mid로 두고 자를 길이에 대한 반복문을 통해서 얻은 나무의 길이와 필요한 나무의 길이를 비교해서 다시 이분 탐색을 진행한다.

댓글남기기