문제
농부 존은 소들의 저녁식사 줄 세우는 새로운 방법을 개발 했다. N(1~15)마리의 소들을 순서대로 세 워놓은 후, 각 소들 사이에 +, - , . 셋 중 1가지가 써져있는 냅킨을 배치해서 최종 결과가 0 이 되게 해야 하는 것이다. 점(.)이 써져있는 냅킨을 통해 더 큰 수를 만들 수 있게 된다. 아래와 같은 경우를 보자. (ps .이 써져있는 냅킨은 '공백'이라고 생각하면 된다.) 1-2.3-4.5+6.7 이와 같은 배치는 1-23-45+67 을 나타낸다. 결과는 0 이다. 10.11은 1011 로 해석된다.
입력
첫 째 줄에는 소들의 수 N(1 ≤ N ≤ 15)이 입력된다.
출력
처음 20줄에 대해 가능한 20가지 답을 출력하는데, 사전 순으로 앞선 것을 출력한다. 순서는 +가 가장 앞서고 -와 . 이 순서대로 뒤따른다. 답이 20개 이하라면 가능한 답을 각 숫자와 문자 사이에 공백을 두고 출력한다. 모두 출력한다. 20개를 초과하는 경우 21번째 답부터는 출력하지 않는다. 마지막 줄에는 가능한 답의 총 가지수를 출력한다.
예제 입력
7
예제 출력
1 + 2 - 3 + 4 - 5 - 6 + 7
1 + 2 - 3 - 4 + 5 + 6 - 7
1 - 2 + 3 + 4 - 5 + 6 - 7
1 - 2 - 3 - 4 - 5 + 6 + 7
1 - 2 . 3 + 4 + 5 + 6 + 7
1 - 2 . 3 - 4 . 5 + 6 . 7
6
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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
|
//모든 조합을 사용해서 푸는 "백트래킹을 활용한 완전 탐색 문제"
//+ - . 순으로 모든 조합을 계산해보기...
// . 처리 하는 것이 핵심! (연속되는 .이 나오면...?)
// 나머지는 그렇게 어렵지 않음..
// 시간복잡도는 3의 15승 안에 모든 것이 끝남..
#include <stdio.h>
int totalCount = 0;
//현재위치, n값, 현재까지 Sum값, 부호를 저장할 배열, 단순 숫자 배열, 단순 부호 배열.
void dessert(int x, int n, int sum, int temp[], int arr[], int ine[]){
//기저조건
if(x==n){
//총 합
int result = arr[0];
// . 이 연속되었는지 갯수를 세어주는 Count
int count = 0;
int countSum = 0;
int countTemp = 1;
for(int i=0; i<n; i++){
//
//printf("%d ",arr[i]);
//printf("%c ",temp[i]);
//
if(i==0 && temp[i]==46){
result -= 1;
}
if(temp[i-1]==46 && temp[i] !=46){
result += countSum;
count = 0;
countSum = 0;
}
if(temp[i]==43 && temp[i+1]!=46){
result += arr[i+1];
}else if(temp[i]==45 && temp[i+1]!=46){
result -= arr[i+1];
}else if(temp[i]==46){
if(count == 0){
if(i!=0 && temp[i-1]==43){
countTemp = 1;
}else if(temp[i-1]==45){
countTemp = -1;
}
if(arr[i]>=9){
countSum = (arr[i]*100 + arr[i+1])*countTemp;
}else{
countSum = (arr[i]*10 + arr[i+1])*countTemp;
}
count++;
}else{
if(arr[i]>=9){
countSum = countSum * 100 + (arr[i+1]*countTemp);
}else{
countSum = countSum * 10 + (arr[i+1]*countTemp);
}
count++;
}
if(i==n-1){
result += countSum;
}
if(count==n){
result -= 1;
}
}
}
//
//printf("%d",arr[n]);
//printf(" = %d",result);
//printf("\n");
//
if(result == 0){
if(totalCount<20){
for(int i=0; i<n; i++){
printf("%d ",arr[i]);
printf("%c ",temp[i]);
}
printf("%d\n",arr[n]);
}
totalCount++;
}
return;
}else{
for(int i=0; i<3; i++){
temp[x] = ine[i];
dessert(x+1, n, 0, temp, arr, ine);
}
return;
}
}
int main() {
//입력 크기 초기화
int n;
scanf("%d",&n);
//현재까지의 값
int sum = 0;
//입력 란에 따른 숫자 배열 초기화
int arr[25];
int temp[25];
for(int i=0; i<n; i++){
arr[i] = i+1;
}
//부호 배열 초기화
int ine[3];
ine[0] = '+';
ine[1] = '-';
ine[2] = '.';
dessert(0, n-1, 0, temp, arr, ine);
//if(totalCount<20){
// printf("%d",totalCount);
//}
printf("%d",totalCount);
return 0;
}
|
'[C++] 알고리즘 교육 > 7~8. 재귀함수' 카테고리의 다른 글
[알고리즘 8.1.5] 재귀함수 - inequal (0) | 2019.04.26 |
---|---|
[알고리즘 8.1.3] 재귀함수 - division (0) | 2019.04.26 |
[알고리즘 8.1.2] 재귀함수 - tobin (0) | 2019.04.25 |
[알고리즘 8.1.1] 재귀함수 - 순열 구하기 (0) | 2019.04.25 |
[알고리즘 7.1.3] 재귀함수 - mountain (0) | 2019.04.25 |
댓글