이 문제의 경우 최적의 값은 5 -> 7 -> 9로
21이 최적의 값이지만
매 상황에서 가장 큰 값만 고른다면(그리디)
5 -> 10 -> 4로 19가 나온다.
(결론)
일반적인 상황에서 그리디 알고리즘은 최적의 해를 보장하지 않는다. 하지만 코딩테스트에서는 입력값과 출력값을 정하는 경우가 많아 그리디 알고리즘으로 분류되는 문제는 단순히 그리디 알고리즘으로 풀어도 최적의 해가 나오는 경우가 많다.
손님에게 1260원을 거슬러줘야한다고 하면
가장 큰 단위의 화폐부터 차례로 거슬러준다면 최적의 해이다.
왜 그런지 정당성 분석을 해보자.
-> 가지고 있는 동전 중에서 큰 단위가 항상 작은 단위의 배수이므로 ( 500원 100원 50원 10원 ) 작은 단위의 동전을 종합해 다른 해가 나올수 없기 때문에다.
만약?
800원을 거슬러 줘야하는데 화폐 단위가 500,400,100이면?
500이 400원의 배수가 아니다 조합을 통해 다른 해가 나올 수 있다.
800원
500원 100원 100원 100원 -> 4개
400원 400원 -> 2개
따라서 이 문제는 그리디 알고리즘으로 풀 수 없는 문제이다 (왜? 최적의 해를 보장하지 않아서 <- 정당성 분석)
n, m, k = map(int, input().split())
data = list(map(int, input().split()))
result = 0
data.sort(reverse=True)
a = m // (k + 1) * k + (m % (k + 1))
result += data[0] * a
result += data[1] * (m - a)
print(result)
n, m, k = map(int, input().split())
data = list(map(int, input().split()))
result = 0
data.sort(reverse=True)
a = m // k
b = m % k
result += data[0] * k * a
result += data[1] * b
print(result)
입력값이
5 8 3
2 4 5 4 6
이라고 하면
6 6 6 5 6 6 6 5 이렇게 더해야한다.
k가 3이기 때문에 6 3번 반복 5 한번 또 6 3 반복 5
이렇게 나오는걸 보고 8을 3으로 나눈값의 몫만큼 가장 큰수를 반복하고 나머지만큼 두번째로 큰수가 더해지는구나 라고 생각했다
1)
하지만 만약
5 9 3
2 4 5 4 6
라면 m이 k에 나누어 떨어진다면 큰수만 9번 반복하게 된다.
위에 코드가 출력값을 잘 뱉은 이유는 그냥 운이였다....
(입력 출력 예시값이 더 있다면 좋았을텐데)
2)
5 8 5
2 4 5 4 6
이것도 8/5 는 1
6 6 6 6 6 5 5 5 이렇게 더해진다...
6 6 6 6 6 5 6 6 (원래라면)
위에 두 문제점을 고려해 다시 생각해봤다.
1)
5 8 3
2 4 5 4 6
6 6 6 5 // 이렇게 하나의 그룹으로 치환해서 생각해보자
6k(8/4) = 6 6 6 6 6 6
5*2= 5 5
2)
5 9 3
2 4 5 4 6
6 6 6 5 6 6 6 5 6
63(9/4) = 6 6 6 6 6 6 (6번)
5*(9/4)=5 5 (2번)
총 8번으로 6이 한번 더 반복되어야한다.
그렇다면 나머지를 이용하면 어떨까?
9%4=1 을 이용해
3(k)*(9/4)+(9%4) = 7
3)
5 8 5
2 4 5 4 6
6(5(k)(8/6)+(8%6))= 6 6 6 6 6 /5/ 6 6
5*(8/6)= 5
n, m = map(int, input().split())
result = []
for i in range(n):
result.append(min(list(map(int, input().split()))))
print(max(result))
각 행을 list로 받을때마다 최소값을 result에 보관
result중 가장 큰 값을 찾음
n, k = map(int, input().split())
result = 0
while (n >= k):
if (n // k > 0):
result += n % k
n = n // k
result += 1
while n > 1:
if n == 1:
break
n = n - 1
result += 1
print(result)
1)
25 5
25 -> 5 -> 1
2)
25 3
25 -> 24 -> 8 -> 7 -> 6 -> 2 -> 1
25/3 = 8 25%3 = 1
8/3=2 8%3=2
2-1=1
우선 나누고 나머지를 더한다
그리고 나눠야하는 값이 k 보다 작으면 1이 될때까지 1을 뺀다.
참고 문서
이것이 코딩테스트다 - 나동빈
안녕하세요! 😊입니다~ 이렇게 문제점들을 찾아서 써보면 해결하는 데에도 도움이 되고 나중에 보기도 좋을 것 같아요! 더 노력하겠습니다💪 저는 벨로그가 처음이라 아직 미숙한 부분들이 많은데 나중에 궁금한 부분들 여쭤볼게요!!
안녕하세요 알고리줌님! 파파이썬입니다~
해결 과정을 모두 적어주시니 읽는 저도 이해가 참 잘되네요!
코드만 올려놓았는데 저도 그렇게 작성을 해보아야겠어요:)
많이 배워갑니다! 내일도 파이팅입니다!