길이가 N인 수식이 있다. 수식은 0보다 크거나 같고, 9보다 작거나 같은 정수와 연산자(+, -, ×)로 이루어져 있다. 곱하기의 연산자 우선순위가 더하기와 빼기보다 높기 때문에, 곱하기를 먼저 계산 해야 한다. 수식을 계산할 때는 왼쪽에서부터 순서대로 계산해야 한다. 예를 들어, 3+8×7-9×2의 결과는 41이다.
수식에 괄호를 추가하면, 괄호 안에 들어있는 식은 먼저 계산해야 한다. 단, 괄호 안에는 연산자가 하나만 들어 있어야 한다. 예를 들어, 3+8×7-9×2에 괄호를 (3+8)×7-(9×2)와 같이 추가했으면, 식의 결과는 59가 된다. 하지만, 중첩된 괄호는 사용할 수 없다. 즉, 3+((8×7)-9)×2, 3+((8×7)-(9×2))은 모두 괄호 안에 괄호가 있기 때문에, 올바른 식이 아니다.
수식이 주어졌을 때, 괄호를 적절히 추가해 만들 수 있는 식의 결과의 최댓값을 구하는 프로그램을 작성하시오. 추가하는 괄호 개수의 제한은 없으며, 추가하지 않아도 된다.
첫째 줄에 수식의 길이 N(1 ≤ N ≤ 19)가 주어진다. 둘째 줄에는 수식이 주어진다. 수식에 포함된 정수는 모두 0보다 크거나 같고, 9보다 작거나 같다. 문자열은 정수로 시작하고, 연산자와 정수가 번갈아가면서 나온다. 연산자는 +, -, 중 하나이다. 여기서 는 곱하기 연산을 나타내는 × 연산이다. 항상 올바른 수식만 주어지기 때문에, N은 홀수이다.
첫째 줄에 괄호를 적절히 추가해서 얻을 수 있는 결과의 최댓값을 출력한다. 정답은 231보다 작고, -231보다 크다.
https://www.acmicpc.net/problem/16638
괄호안에는 두개의숫자, 연산기호만이 들어가야하면 이중괄호는 안된다.
구현해야할것이 너무많다. 가장 효율적인 방법이 위의 1~2번과정 이지 않나 싶다.
#define _CRT_SECURE_NO_WARNINGS
#include <bits/stdc++.h>
using std::vector; using std::stack; using std::queue; using std::deque; using std::tuple;
using std::string; using std::pair; using std::make_tuple; using std::tie;
using std::sort; using std::swap;
using std::make_pair; using std::max_element; using std::min_element; using std::bitset; using std::to_string;
using std::fill; using std::priority_queue; using std::greater;
//using std::max; using std::min; using std::map;
typedef long long ll;
typedef pair<int, int> pii;
typedef pair<double, int> pdi;
typedef pair<int, double> pid;
typedef pair<double, double> pdd;
typedef pair<ll, ll> pll;
typedef pair<ll, int> pli;
typedef pair<int, ll> pil;
typedef pair<int, char> pic;
typedef pair<char, int> pci;
typedef pair<string, int> psi;
typedef pair<int, string> pis;
typedef pair<string, string> pss;
int N;
vector<int> num; //0~9
vector<int> op; //1:+ 2:- 3:*
ll max = INT32_MIN; //출력할 최댓값
void init();
int monomialCalculate(int left, int right, int OP);
ll calculate(vector<int>& index);
ll polynomialCalculate(vector<int>& newNUM, vector<int>& newOP);
void debug(int n, vector<int>& a, vector<int>& b, ll r); //논리확인용
void picked(vector<int>& index);
void func();
void init() {
scanf("%d", &N);
char o;
scanf("%c", &o); //줄바꿈문자 제거
for (int i = 0; i < N; i++) {
char o;
scanf("%c", &o);
if (i % 2) //기호
if (o == '+')
op.push_back(1);
else if (o == '-')
op.push_back(2);
else
op.push_back(3);
else //숫자
num.push_back(o - '0');
}
}
//두 숫자를 OP연산기호(숫자로 대체)로 계산한값 리턴
int monomialCalculate(int left, int right, int OP) {
if (OP == 1) //+
return left + right;
else if (OP == 2) //-
return left - right;
else if (OP == 3) //*
return left * right;
}
//괄호를 풀고 전체적인 연산흐름을 맡음
ll calculate(vector<int>& opIndex) {
vector<int> NUM, OP;
//1-1. 괄호안의 수식을 먼저 계산
//left, right : 괄호안 두 숫자. 연산기호를 기준으로 좌,우 두 숫자
bool prevBrace = false;
//전체 연산기호 탐색
for (int i = 0; i < op.size(); i++) {
bool findIndex = false;
//그 중 선택한 연산기호를 탐색
for (int j = 0; j < opIndex.size(); j++) {
if (i == opIndex[j]) { //괄호안 연산기호였다.
findIndex = true;
break;
}
}
//괄호안 연산기호 라면
if (findIndex) {
//NUM에 괄호안을 계산한숫자추가, OP에 연산기호추가X
int left{ num[i] }, right{ num[i + 1] }; //두 숫자
//printf("1>> left:%d right:%d\n", left, right);
NUM.push_back(monomialCalculate(left, right, op[i])); //괄호를 계산해서 NUM에 추가
prevBrace = true;
}
else { //괄호밖 연산기호 라면
if (prevBrace) { //바로직전에 괄호를 계산 했었다면
//NUM에 num[i]숫자추가X, OP에 연산기호추가O
OP.push_back(op[i]);
prevBrace = false; //체크 해제
}
else {
//NUM에 num[i]숫자추가O, OP에 연산기호추가O
NUM.push_back(num[i]);
OP.push_back(op[i]);
}
}
//마지막인덱스이고 괄호가 아니면 마지막 숫자를 NUM에 추가해줘야함
if (i == op.size() - 1 && !findIndex)
NUM.push_back(num[op.size()]);
}
//debug(22, NUM, OP, polynomialCalculate(NUM, OP));
//2. 괄호를 푼 수식을 계산해서 리턴
ll res = polynomialCalculate(NUM, OP);
return res;
}
//전체 수식을 계산하여 리턴
//곱하기를 먼저계산한다.
ll polynomialCalculate(vector<int>& NUM, vector<int>& OP) {
vector<int> A, B;
//복사
for (int i = 0; i < NUM.size(); i++) A.push_back(NUM[i]);
for (int i = 0; i < OP.size(); i++) B.push_back(OP[i]);
//1-2.
while (1) { //곱하기 기호를 발견하지 못할때 까지 반복
bool findMultiple = false;
for (int i = 0; i < B.size(); i++) {
//1-3.
if (B[i] == 3) {
findMultiple = true;
int left{ A[i] }, right{ A[i + 1] };
//printf("3>> left:%d right:%d\n", left, right);
//1-4.
//left위치에 left,right곱한 결과를 덮어씌우고
//right원소(A[i + 1])를 삭제
//B[i] 도 삭제
A[i] = monomialCalculate(left, right, 3); //곱하기 연산실행
A.erase(A.begin() + i + 1); //right숫자 삭제
B.erase(B.begin() + i); //곱연산기호 삭제
}
}
//1-5.
if (!findMultiple)
break;
}
ll res = A[0];
for (int i = 1; i < A.size(); i++)
res = monomialCalculate(res, A[i], B[i - 1]);
return res;
}
//두 벡터의 내용을 출력 - 확인용
void debug(int n, vector<int>& a, vector<int>& b, ll r) {
printf("%d>> ", n);
bool A = true, B = true;
int idxA = 0, idxB = 0;
int lenA = a.size(), lenB = b.size();
while (1) {
if (A)
printf("%d ", a[idxA]);
if (B)
if (b[idxB] == 1)
printf("+ ");
else if (b[idxB] == 2)
printf("- ");
else if (b[idxB] == 3)
printf("* ");
idxA++;
idxB++;
if (idxA == lenA) A = false;
if (idxB == lenB) B = false;
if (!A && !B) break;
}
printf("= %lld\n", r);
}
//괄호를 추가해서 수식을 계산해본다.
//서로 인접하지않도록 연산기호를 선택(괄호를 추가하는것)하는 모든 경우의수에서
//각각 경우마다의 계산결과의 최댓값을 max로 갱신
void picked(vector<int>& opIndex) {
//1. 2개이상 연산기호를 pick할때 각 연산기호가 서로 인접하면 안됨.
bool pass = false;
if (opIndex.size() >= 2) {
//선택한 두개의 연산기호가 서로 인접한지 확인
for (int i = 0; i < opIndex.size(); i++) {
int prevIndex = opIndex[i];
for (int j = i + 1; j < opIndex.size(); j++) {
int nextIndex = opIndex[j];
if (nextIndex - prevIndex == 1) { //서로 인접하다면
pass = true;
break;
}
}
}
}
if (pass) return; //서로 인접한 연산기호를 사용하는 경우는 제외
//2. 현재까지 담은 오퍼레이터만큼 결과를 계산
if (!opIndex.empty()) { //괄호를 추가하지않고 계산하는경우는 아래 func()에서 미리 해두었음
ll res = calculate(opIndex);
if (max < res) max = res;
}
//3. OP의 연산기호를 여럿 선택하는 경우의수를 재귀 방법으로 구현
int smallest = opIndex.empty() ? 0 : opIndex.back() + 1; //index에들어간 인덱스의 바로 다음 인덱스
for (int i = smallest; i < op.size(); i++) {
opIndex.push_back(i);
picked(opIndex);
opIndex.pop_back();
}
}
void func() {
//하나의 연산기호를 pick하는것이 괄호를 만드는것.
//다만, 서로 인접한 연산기호를 선택한다면 괄호를 여러개 만들지 못함.
//서로 인접하지않은 연산기호를 pick 해야한다.
//pick 한뒤 각각 경우마다의 최댓값을 구해본다.
vector<int> opIndex;
ll res = polynomialCalculate(num, op); //괄호를 추가하지않고 계산하는경우
if (max < res) max = res;
picked(opIndex); //괄호를 추가해서 계산하는 경우
printf("%lld", max); //구한 최댓값
}
int main(void) {
init();
func();
return 0;
}