[문제]
N명의 병사가 무작위로 나열되어 있습니다. 각 병사는 특정한 값의 전투력을 보유하고 있으며, 병사를 배치할 때는
전투력이 높은 병사가 앞쪽에 오도록 내림차순으로 배치를 하고자 합니다. 다시 말해 앞쪽에 있는 병사의 전투력이
항상 뒤쪽에 있는 병사보다 높아야 합니다.
또한 배치 과정에서는 특정한 위치에 있는 병사를 열외시키는 방법을 이용합니다. 그러면서도 남아 있는 병사의
수가 최대가 되도록 하고 싶습니다.
예를 들어, N = 7 일 때 나열된 병사들의 전투력이 다음과 같다고 가정하겠습니다.
병사 번호 | 1 | 2 | 3 | 4 | 5 | 6 | 7 |
전투력 | 15 | 11 | 4 | 8 | 5 | 2 | 4 |
이때 3번 병사와 6번 병사를 열외시키면, 다음과 같이 남아 있는 병사의 수가 내림차순의 형태가 되며 5명이 됩니다.
이는 남아 있는 병사의 수가 최대가 되도록 하는 방법입니다.
병사 번호 | 1 | 2 | 4 | 5 | 7 |
전투력 | 15 | 11 | 8 | 5 | 4 |
병사에 대한 정보가 주어졌을 때, 남아 있는 병사의 수가 최대가 되도록 하기 위해서 열외시켜야 하는 병사의 수를
출력하는 프로그램을 작성하세요.
[입력 조건]
1. 첫째 줄에 N이 주어집니다. (1 <= N <= 2,000) 둘째 줄에 각 병사의 전투력이 공백으로 구분되어 차례대로 주어집니다.
각 병사의 전투력은 10,000,000보다 작거나 같은 자연수입니다.
[출력 조건]
첫째 줄에 남아 있는 병사의 수가 최대가 되도록 하기 위해서 열외시켜야 하는 병수의 수를 출력합니다.
<입력 예시>
7
15 11 4 8 5 2 4
<출력 예시>
2
[풀이]
n = int(input())
array = list(map(int, input().split()))
# 순서를 뒤집어 '가장 긴 증가하는 부분 수열' 문제로 변환
array.reverse()
# 다이나믹 프로그래밍을 위한 1차원 DP 테이블 초기화
dp = [1] * n
# 가장 긴 증가하는 부분 수열(LIS) 알고리즘 수행
for i in range(1, n) :
for j in range(0, i) :
if array[j] < array[i] :
dp[i] = max(dp[i], dp[j]+1)
# 열외시켜야 하는 병사의 최소 수를 출력
print(n - max(dp))
* '가장 긴 증가하는 부분 수열' : 하나의 수열이 주어졌을 때 값들이 증가하는 형태의 가장 긴 부분 수열을 찾는 문제이다.
예를 들어 하나의 수열 array = {10, 20, 10, 30, 20, 50}가 있다고 하자. 이때 가장 긴 증가하는 부분 수열은
{10, 20, 30, 50}이 될 것이다. 'D[i] = array[i]를 마지막 원소로 가지는 부분 수열의 최대 길이'라고 정의하면,
가장 긴 증가하는 부분 수열을 계산하는 점화식은 다음과 같다.
모든 0 <= j < i 에 대하여, D[i] = max(D[i], D[j] + 1) if array[j] < array[i]
출처
이것이 코딩 테스트다 with 파이썬 - 나동빈 저
'알고리즘 > 이코테 알고리즘 유형별 기출문제' 카테고리의 다른 글
[다이나믹 프로그래밍] 이코테 (파이썬) 편집 거리 풀이 (0) | 2022.01.13 |
---|---|
[다이나믹 프로그래밍] 이코테 (파이썬) 못생긴 수 풀이 (0) | 2022.01.13 |
[다이나믹 프로그래밍] 이코테 (파이썬) 퇴사 풀이 (0) | 2022.01.12 |
[다이나믹 프로그래밍] 이코테 (파이썬) 정수 삼각형 풀이 (0) | 2022.01.12 |
[다이나믹 프로그래밍] 이코테 (파이썬) 금광 풀이 (0) | 2022.01.12 |