Algorithm
(BOJ) 2504 풀이
snwo
2022. 2. 3. 19:22
#include <bits/stdc++.h>
using namespace std;
int main(void)
{
ios::sync_with_stdio(0);
cin.tie(0);
string input;
cin >> input;
stack<char> S;
int flag = 0, ans = 0, tmp = 1;
for (int i = 0; i < input.size(); i++)
{
if (input[i] == '(')
{
tmp *= 2;
S.push(input[i]);
}
if (input[i] == '[')
{
tmp *= 3;
S.push(input[i]);
}
if (input[i] == ')')
{
if (input[i - 1] == '(') // 2
{
S.pop();
ans += tmp;
}
else if (!S.empty() && S.top() == '(')
{
S.pop();
}
else
{
flag = 1;
break;
}
tmp /= 2;
}
else if (input[i] == ']')
{
if (input[i - 1] == '[') // 3
{
S.pop();
ans += tmp;
}
else if (!S.empty() && S.top() == '[')
{
S.pop();
}
else
{
flag = 1;
break;
}
tmp /= 3;
}
}
if (!S.empty())
{
flag = 1;
}
if (!flag)
cout << ans;
else
cout << 0;
}
이해하는데에 조금 걸린 문제였다.
[]
→ 3
()
→ 2
[x]
→ x*3
(x)
→ x*2
(x,y,z)
→ x+y+z
(값)(값)
→ 값+값
위 규칙을 이용해서 괄호값을 계산해야한다.
여는 괄호가 나타날 때마다 tmp에 2 or 3 을 곱한 뒤,
닫는 괄호가 나타났을 때, 직전 괄호와 쌍이 맞으면 ans 에 더해줬다.
이후 tmp 를 2 or 3 으로 나눈다.
직전 괄호와 쌍이 안맞고, 스택에 있는 괄호와도 안맞으면 잘못된 괄호이므로 flag 변수를 1로 세팅하고 종료.
(()[[]])([])
위와 같은 예시가 있을 때,
(()
: 일단 여기까지 진행했을 때 tmp 변수는 4고, 이 값을 ans 에 저장한 후, 소괄호를 빠져나왔으니 tmp 를 2로 나눈다.
올바른 괄호라면, 안쪽에 뭐가 있든 일단 이 괄호의 최솟값은 4가 된다. (())
.
[[]])
: 현재 tmp 가 2 이므로, 2*3*3
이 ans 에 더해진다. 2가 곱해진 이유는, 대괄호 두 쌍이 소괄호 안에 있기 때문이다.
위 계산형식은, (2+6) → (2+6) * 2
가 아니라 2*2 + 6*2
이다.
더한 뒤 2를 곱하는게 아니라 괄호값을 미리 곱한뒤 ans 에서 더해진다. ( 곱셈 분배법칙 )
2*2+3*3*2+3*2
이런느낌으로