문제
N개의 수로 이루어진 수열 A1, A2, ..., AN이 주어진다. 또, 수와 수 사이에 끼워넣을 수 있는 N-1개의 연산자가 주어진다. 연산자는 덧셈(+), 뺄셈(-), 곱셈(×), 나눗셈(÷)으로만 이루어져 있다.
우리는 수와 수 사이에 연산자를 하나씩 넣어서, 수식을 하나 만들 수 있다. 이때, 주어진 수의 순서를 바꾸면 안 된다.
예를 들어, 6개의 수로 이루어진 수열이 1, 2, 3, 4, 5, 6이고, 주어진 연산자가 덧셈(+) 2개, 뺄셈(-) 1개, 곱셈(×) 1개, 나눗셈(÷) 1개인 경우에는 총 60가지의 식을 만들 수 있다. 예를 들어, 아래와 같은 식을 만들 수 있다.
- 1+2+3-4×5÷6
- 1÷2+3+4-5×6
- 1+2÷3×4-5+6
- 1÷2×3-4+5+6
식의 계산은 연산자 우선 순위를 무시하고 앞에서부터 진행해야 한다. 또, 나눗셈은 정수 나눗셈으로 몫만 취한다. 음수를 양수로 나눌 때는 C++14의 기준을 따른다. 즉, 양수로 바꾼 뒤 몫을 취하고, 그 몫을 음수로 바꾼 것과 같다. 이에 따라서, 위의 식 4개의 결과를 계산해보면 아래와 같다.
- 1+2+3-4×5÷6 = 1
- 1÷2+3+4-5×6 = 12
- 1+2÷3×4-5+6 = 5
- 1÷2×3-4+5+6 = 7
N개의 수와 N-1개의 연산자가 주어졌을 때, 만들 수 있는 식의 결과가 최대인 것과 최소인 것을 구하는 프로그램을 작성하시오.
입력
첫째 줄에 수의 개수 N(2 ≤ N ≤ 11)가 주어진다. 둘째 줄에는 A1, A2, ..., AN이 주어진다. (1 ≤ Ai ≤ 100) 셋째 줄에는 합이 N-1인 4개의 정수가 주어지는데, 차례대로 덧셈(+)의 개수, 뺄셈(-)의 개수, 곱셈(×)의 개수, 나눗셈(÷)의 개수이다.
출력
첫째 줄에 만들 수 있는 식의 결과의 최댓값을, 둘째 줄에는 최솟값을 출력한다. 최댓값과 최솟값이 항상 -10억보다 크거나 같고, 10억보다 작거나 같은 결과가 나오는 입력만 주어진다. 또한, 앞에서부터 계산했을 때, 중간에 계산되는 식의 결과도 항상 -10억보다 크거나 같고, 10억보다 작거나 같다.
내 풀이
입력받은 연산자에 대한 순열을 만들어서 모든 경우의 수에 대해 계산해보고 그 중 최솟값과 최댓값을 구하면 되는 문제
순열은 itertools의 permutations를 사용했는데, 직접 코딩할 줄도 알아야 한다.
처음에는 eval() 함수를 사용해서 단순히 계산했는데, 예제와 출력이 달라서 문제를 다시 읽어보니 연산자 우선순위를 무시하고 계산하는 문제였다. 코드를 조금 손보고 제출했더니 합격!
아래 코드의 주석에 좀더 자세한 설명이 있다.
점점 더 내 힘으로만 풀 수 있는 문제가 많아지는 것 같아 기쁘다.
문제를 읽고 이해를 먼저 한 후에 푸는 것도 좋지만,
일단 입력을 받는 코드부터 (이건 많이 해서 익숙해졌으니까) 코딩하면서 순차적으로 어떻게 문제를 풀어나가야 할지 주석으로 써본 후에 코딩을 하는 방식이 나에게는 맞는 것 같다!
그리고 그 편이 좀더 '일단 돌아가는 코드 만들기'에 쉬운 것 같다.
일단 돌아가는 코드를 만들어야 그 후에 효율성을 기해 볼 수 있다. 그러니 일단 시도해 볼 것!
순열을 구현하는 알고리즘은 이곳에서 볼 수 있다.
'알고리즘' 카테고리의 다른 글
BOJ 14890 경사로 (0) | 2019.09.18 |
---|---|
파이썬으로 순열 구현하기 (0) | 2019.09.14 |
BOJ 14503 로봇청소기 (0) | 2019.09.14 |
BOJ 14502 연구소 (0) | 2019.09.13 |
BOJ 14501 퇴사 (0) | 2019.09.13 |