-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path10835.cpp
51 lines (44 loc) · 2.5 KB
/
10835.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
#include <cstdio> // scanf, printf.
#include <vector> // 인접 리스트.
#include <string.h> // memset. 배열 초기화 함수.
#include <algorithm> // max. 둘 중 최대값 리턴.
using namespace std;
int N; // 한 더미의 카드의 개수를 나타내는 자연수. 최대 2,000.
vector<int> left; // 왼쪽 더미.
vector<int> right; // 오른쪽 더미.
int memo[2001][2001]; // 메모이제이션 변수. 2차원.
int solve(int L, int R) // 함수의 정의 : 최종 점수의 최대값 출력.
{
// 기저 조건. 바닥 조건. 종료 조건.
if( L == N || R == N )
// 규칙 3. 어느 쪽 더미든 남은 카드가 없다면
return 0; // 게임 끝. 더 이상 얻는 점수 없음.
int &ret = memo[L][R]; // 주소연산자로 연결. #define ret memo[L][R]; 과 똑같이 작동한다.
if( ret != -1 ) // solve(L,R)이 이미 실행되어서 그 결과값이 있다면,
return ret; // 그 결과값 리턴.
// 메모이제이션을 통해 메모리를 희생하는 대신 중복 호출을 줄여 처리 시간을 줄인다.
ret = max(solve(L+1,R), solve(L+1,R+1)); // 둘 중 최대값을 ret에 입력.
// 규칙 1. 언제든지 왼쪽 카드만 통에 버릴 수도 있고 // solve(L+1,R)
// 왼쪽 카드와 오른쪽 카드를 둘 다 통에 버릴 수도 있다. // solve(L+1,R+1)
// 이때 얻는 점수 없다.
if(left[L] > right[R])
// 규칙 2. 오른쪽 카드에 적힌 수가 왼쪽 카드에 적힌 수보다 작은 경우에는
ret = max(ret, solve(L,R+1)+right[R]); // 둘 중 최대값을 ret에 입력.
// 오른쪽 카드만 통에 버릴 수도 있다. // solve(L,R+1)
// 이 때 오른쪽 카드에 적힌 수만큼 점수를 얻는다. // +right[R]
return ret; // 최종 점수의 최대값 리턴.
}
int main()
{
scanf("%d",&N); // 한 더미의 카드의 개수 입력.
left.resize(N); // left 리스트 크기 N만큼(0~N-1) 증가.
right.resize(N); // right 리스트 크기 N만큼(0~N-1) 증가.
memset(memo,-1,sizeof(memo)); // 메모이제이션 변수 배열 -1로 모두 초기화.
// -1은 절대 점수로 나올 수 없는 값이기 때문에 초기화 값으로 지정했다.
for(int i=0;i<N;i++)
scanf("%d",&left[i]); // 왼쪽 더미 카드 순서대로 입력.
for(int i=0;i<N;i++)
scanf("%d",&right[i]); // 오른쪽 더미 카드 순서대로 입력.
printf("%d",solve(0,0)); // 최종 점수의 최대값 출력.
// solve(0,0) 은 왼쪽, 오른쪽 더미 모두 첫번재 카드부터 시작한다는 뜻.
}