블로그를 이전 중이라 완료되기 전까지는 벨로그에 작성할 계획입니다.
이후 모든 글은 https://weekwith.me 에 작성 예정이니 다른 글이 궁금하시다면 해당 링크를 통해 방문해주세요.또한 본 글의 모든 내용은 한빛미디어 출판사에서 출판한 나동빈의 이것이 취업을 위한 코딩 테스트다 with 파이썬 : 그리디를 참고했음을 알립니다.
그리디(Greedy) 알고리즘의 개념에 대해 학습하고 관련된 문제를 풀이한다.
이름에서 알 수 있듯이 어떠한 문제가 있을 때 단순 무식하게, 탐욕적으로 문제를 푸는 알고리즘이다. 여기서 탐욕적이란 단어의 의미는 현재 상황에서 지금 당장 좋은 것만 고르는 방법을 의미하기 때문에 현재의 선택이 이후에 끼칠 영향에 대해서는 고민하지 않고 매 순간 가장 좋아 보이는 것만 선택한다.
기준에 따라 가장 좋은 것을 선택하는 알고리즘이기 때문에 문제에서 정렬 과 관련된 기준이 제시된다. 따라서 정렬 알고리즘과 짝을 이루어 출제되는 경우가 대다수다. 이외에도 다익스트라(Dijkstra) 알고리즘이나 크루스칼(Kruskal) 알고리즘 또한 그리디 알고리즘에 해당된다.
가장 큰 수를 최대 반복할 수 있는 만큼 반복한 이후 그 다음으로 큰 수를 하나씩 섞어주면 된다. 따라서 내장함수 sort()
를 사용해 내림차순 정렬한 뒤 배열의 첫 번째 요소와 두 번째 요소를 각각 변수에 할당한다.
결과적으로 가장 큰 수를 최대 K번 반복하고 그 다음 수 1번 반복하는 (K + 1) 개의 반복적인 형태로 구성되기 때문에 M을 해당 (K + 1) 수로 나누어 생긴 몫을 (K + 1)의 반복횟수, 나머지를 가장 큰 수의 반복 횟수로 가져가면 된다.
N, M, K = list(map(int, input().split(" ")))
numbers: list[int] = list(map(int, input().split(" ")))
numbers.sort(reverse=True)
largest_number: int = numbers[0]
second_number: int = numbers[1]
set_count = M // (K + 1)
remained_count = M % (K + 1)
answer: int = (
(largest_number * K + second_number) * set_count
+
largest_number * remained_count
)
print(answer)
내림차순 정렬을 수행하는 내장함수 sort()
의 시간 복잡도가 O(NlogN)이기 때문에 결론적으로 O(NlogN)이다. 나머지 연산의 경우 인덱스를 통한 배열의 요소 조회 및 사칙 연산일 뿐이기 때문에 상수 시간이 걸린다.
이중 반복문을 돌면서 행마다 가장 작은 수를 구하고 그 중 가장 큰 수를 출력하면 된다.
N, M = list(map(int, input().split()))
answer: int = 0
for _ in range(N):
smallest_card: int = 10000
for card in list(map(int, input().split())):
if card < smallest_card:
smallest_card = card
if answer < smallest_card:
answer = smallest_card
print(answer)
N번 반복문이 도는 내부에 M번의 반복문이 추가적으로 돌기 때문에 시간 복잡도는 O(NM)이다.
주어진 N이 K로 나누어 떨어질 때까지 1씩 빼다가 나누어 떨어지면 나누면 된다.
N, K = list(map(int, input().split()))
answer: int = 0
while N > 1:
if not N % K:
answer += 1
N //= K
else:
answer += 1
N -= 1
print(answer)
위와 같이 계산할 경우 주어진 입력 값의 범위 내에서는 상관 없지만 입력 값의 범위가 더 커지면 하나씩 빼기 때문에 느리게 작동한다.
따라서 아래 코드와 같이 하나씩 빼주던 경우의 수를 한번에 계산하여 조금 더 최적화 해줄 수 있다.
주어진 N 자체를 K로 나누고 그 나머지 몫과 함께 나누는 횟수까지 포함하여 N이 K보다 작아질 때까지 더하게 된다.
N이 K보다 작아지는 순간 더이상 K로 나눌 수 없다는 걸 의미하기 때문에 1이 되기 전까지의 수인 (N - 1) 만큼을 최종적으로 횟수에 더해준다.
N, K = list(map(int, input().split()))
answer: int = 0
while N >= K:
answer += (N % K) + 1
N //= K
answer += (N - 1)
print(answer)
나눌 때마다 계산해야 할 경우의 수가 어림잡아 반씩 줄어들기 때문에 시간 복잡도는 O(logN)이다.
공포도의 경우 동일한 공포도 값이 배수로 존재해야 그룹이 하나씩 증가하게 된다.
예를 들어 공포도 값이 2일 경우 2인 모험가가 4명 있으면 그룹이 두 개가 만들어지는 것이다.
다른 공포도를 섞을 경우 해당 공포도보다 값이 적으면 상관 없지만 해당 공포도보다 값이 크면 결국 그 값에 맞춰 그룹을 만들어야 하기 때문에 동일한 공포도끼리 묶어 그 개수를 공포도로 나누었을 때의 정수 몫이 곧 모험가의 그룹 수에 더해져야 한다.
from collections import defaultdict
N: int = int(input())
F: list(int) = list(map(int, input().split()))
fearnesses: dict[int, int] = defaultdict(int)
for fearness in F:
fearnesses[fearness] += 1
answer: int = 0
for fearness, count in fearnesses.items():
answer += (count // fearness)
print(answer)
배열을 한번 돌면서 딕셔너리 자료형에 그 개수를 만드는 것과 마지막에 그 개수에 따라 모험가의 수를 구하는 것 자체가 동일한 수의 반복문을 수행하기 때문에 모험가의 수인 N만큼의 시간 복잡도인 O(N)이다.
0과 1을 제외한 나머지 수는 곱해야 가장 큰 수가 될 수 있다.
따라서 인덱스 1번부터 반복문을 돌며 누적된 값이 0 또는 1이거나 요소 본인 자체가 0 또는 1일 경우 더하고 나머지는 곱하면 된다.
S: str = input()
answer: int = int(S[0])
for idx in range(1, len(S)):
if (answer <= 1) or (int(S[idx]) <= 1):
answer += int(S[idx])
else:
answer *= int(S[idx])
print(answer)
문자열 S의 길이 만큼 반복문을 수행하면 되기 때문에 시간 복잡도는 O(N)이다.
첫 시작 문자와 다를 때부터 개수를 세는데 이때 연속된 동일한 문자는 한번에 뒤집기 때문에 이전 수와 다를 경우만 변해야 한다.
S: str = input()
answer: int = 0
first_num, previous_num = S[0], S[0]
for num in S:
if (num != first_num) and (num != previous_num):
previous_num = num
answer += 1
elif num != previous_num:
previous_num = num
print(answer)
문자열 S의 길이 만큼 반복문을 수행하면 되기 때문에 시간 복잡도는 O(N)이다.
고민을 하다가 브루트 포스인 것 같은 방법 밖에 떠오르지 않아 우선은 그렇게 풀었다. 아마 시간초과가 될 것이다.
만들 수 있는 수의 조합을 구한 다음 그 개수를 세서 다시 1부터 만들 수 있는 가장 큰 수보다 하나 큰 값까지 반복문을 돌아 만약 해당 조합이 만들 수 없는 경우면 출력하는 방식으로 접근했다.
N: int = int(input())
coins: list[int] = list(map(int, input().split()))
max_price: int = sum(coins)
possible_price: dict[int, int] = { number: 0 for number in range(1, max_price + 1) }
for i in range(len(coins)):
for j in range(i + 1, len(coins) + 1):
possible_price[sum(coins[i:j])] += 1
for price in range(1, max_price + 2):
if not possible_price[price]:
print(price)
break
아래는 책에 나온 모범 답안이었다.
1부터 시작하는 목표값을 설정한 뒤 내림차순으로 정렬된 동전을 하나씩 더해서 만약에 해당 목표값이 동전보다 작을 경우 그 수를 만들 수 없는 것이기에 출력하고 반복문을 멈춘다.
예를 들어서 예시 입력값인 [3, 2, 1, 1, 9]
의 경우 내림차순 정렬하여 [1, 1, 2, 3, 9]
가 되고 1
부터 차례로 목표값인 1
과 비교하고 만약 목표값이 동전보다 작지 않을 경우 해당 동전을 목표값에 더하게 되는데 3
까지 더해진 직후인 목표값 8
이 다음 동전 9
보다 작기 때문에 8
보다 작은 동전은 존재하는 동전들로 만들 수 있다는 의미가 되고 그 다음부터는 9
이상이기 때문에 8
이 곧 만들 수 없는 최소가 된다.
목표값의 시작을 1
로 하기 때문에 주어진 동전의 차례 합보다 무조건적으로 하나씩 커져서 이를 비교하게 된다. 따라서 모든 합을 다 비교하더라도 결국 최대로 구할 수 있는 값보다 1
크기 때문에 오류가 발생하지 않는다.
N: int = int(input())
coins: list[int] = list(map(int, input().split()))
coins.sort()
target: int = 1
for coin in coins:
if target < coin:
break
target += coin
print(target)
처음 생각한 접근법으로 문제를 풀 경우 시간 복잡도는 O(N^2)이다.
책에 나온 모범 답안으로 접근하여 문제를 풀 경우 주어진 N개의 동전 개수 만큼만 반복문을 수행하면 되기 때문에 O(N), 오름차순 정렬을 위해 사용한 내장함수 sort()
의 시간 복잡도가 O(NlogN)이기 때문에 결국 O(NlogN)이다.
각 볼링공의 무게는 전체 볼링공 개수에서 해당 볼링공을 포함하여 동일한 무게를 가진 볼링공의 개수 만큼 빼어 조합을 할 수 있다. 따라서 해시 테이블 -딕셔너리- 을 하나 만들어서 각 무게를 키로, 개수를 값으로 저장한 다음 전체 개수 N에서 각 개수 만큼 뺀걸 해당 개수 만큼 곱해주고 이것이 모든 조합이기 때문에 중복되는 경우를 빼주기 위해 절반으로 나눠주면 된다.
def solution() -> None:
from collections import defaultdict
N, M = list(map(int, input().split()))
balls: dict[int, int] = defaultdict(int)
for ball in list(map(int, input().split())):
balls[ball] += 1
answer: int = 0
for ball_count in balls.values():
answer += ((N - ball_count) * ball_count)
print(answer // 2)
최악의 경우 모든 볼링공의 무게가 전부 달라 하나씩만 존재하는 경우이기 때문에 결국 시간 복잡도는 반복문을 볼링공의 수인 N만큼만 작동해서 O(N)이다.
무지의 먹방 라이브의 경우 30분에 걸쳐 두번 시도했지만 결국 풀지 못해서 답지를 보니 우선순위 큐에 대한 이야기가 나왔다. 그래서 관련 개념을 학습한 뒤에 해당 내용을 다시 도전하기로 하였다.