[백준 C++] 16638 괄호 추가하기 2

이성훈·2022년 9월 30일
0

백준(Baekjoon online judge)

목록 보기
115/177

문제

길이가 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. 두 벡터에 숫자만, 연산기호만 담아놓는다.(NUM, OP)
    문제에서 곱연산이 우선순위를 두었으므로,
    숫자는 항상 연산기호 보다 1개더 많은 상태이므로(올바른 수식에서)
    연산기호의 인덱스 i가있으면 좌우 숫자는 i, i+1의 인덱스를 가짐을 알 수 있다.
    1-1. 괄호를 먼저 연산
    1-2. 남은 연산기호를 전부 탐색
    1-3. 곱연산기호를 만나면 좌우 숫자를 곱연산
    1-4. 그리고 NUM[i]에 계산값을 넣고 NUM[i+1]을 erase
    1-5. 1-2부터 반복하되 곱연산기호를 못찾았으면 아래 2번으로
  3. 곱연산이 없어진 전체 수식을 앞에서부터 뒤로 계산하여 최댓값이면 갱신

구현해야할것이 너무많다. 가장 효율적인 방법이 위의 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;
}
profile
I will be a socially developer

0개의 댓글