본문 바로가기
알고리즘/백준

[백준 11899] 괄호 끼워넣기 (C++)

by fortissimo 2024. 5. 4.

https://www.acmicpc.net/problem/11899

 

문제


심심한 승현이는 너무 심심한 나머지 올바른 괄호열을 가지고 놀고 있었습니다.

(()(()))()()

그러다가 어쩌다 보니 괄호열을 부러뜨렸습니다.

(()    ((    )))()    ()

크게 낙담한 승현이는 노력해 보았지만, 대부분이 부러져 버려 단 한 부분만 재사용할 수 있다는 것을 깨닫게 되었습니다.

)))()

승현이는 이 괄호열을 가지고 놀려고 했으나 올바른 괄호열이 아니기 때문에 행복하지 않았습니다. 이를 보던 지학이는 승현이에게 “그러면 앞과 뒤에 적절하게 괄호를 붙이면 올바른 괄호열이 되지 않을까?”라고 했고, 승현이는 조금 생각한 뒤 그렇게 하기로 했습니다. 예를 들어, 위의 올바르지 않은 괄호열의 경우 앞에 여는 괄호 3개를 붙이면 올바른 괄호열이 됩니다.

((()))()

그러나 괄호열을 사서 붙이는 데에는 돈이 들고 많이 붙일수록 놀기가 불편해지기 때문에, 승현이는 가능한 한 괄호열을 적게 추가하려고 합니다.

승현이가 망가뜨리고 사용 가능한 올바르지 않은 괄호열이 주어질 때, 올바른 괄호열으로 만들기 위해 앞과 뒤에 붙여야 할 괄호의 최소 개수를 구하는 프로그램을 작성하세요.

 

입력


첫 번째 줄에 올바르지 않은 괄호열 S가 주어집니다. S의 길이는 1 이상 50 이하입니다.

 

출력


첫 번째 줄에 S를 올바른 괄호열으로 만들기 위해 앞과 뒤에 붙여야 할 괄호의 최소 개수를 출력합니다. 불가능한 경우는 주어지지 않습니다.

 

문제 풀이


스택을 이용한 문제.

불가능한 경우가 주어지지 않으므로 인덱스 i가  ( 라면 그 이전까지는 올바른 괄호열이거나 아직 닫히지 않은 괄호들이 주어져 있다. 스택에 (를 저장하여 나중에 )를 만나 pop될 수 있도록 한다.

인덱스 i가 ) 라면 스택에 저장된 (와 만나 pop될 수 있다. 만약 스택의 크기가 0이라면 (가 빠진 것이므로 answer의 값을 1 증가시킨다. 스택의 크기가 0이 아니라면 (와 만나 pop되어 정상적인 문자열 ()를 만든다. 스택에는 (만 저장되고, 짝을 만나지 못하는 )는 스택에 저장되지 않으므로 스택의 크기가 0이 아니라면 무조건 (와 만나 정상적인 문자열 ()을 만들 수 있다.

문자열 끝까지 탐색했을 때 스택에 남아 있는 (가 있을 수 있다. 이 수만큼 )가 있어야 문자열 전체가 정상적인 문자열이 된다. 해당 수만큼 answer의 값을 증가시킨다.

 

아래는 코드.

더보기
#include <iostream>
#include <stack>
using namespace std;

int main()
{
	cin.tie(NULL);
	ios::sync_with_stdio(false);

	string str;
	cin >> str;
	stack<char> s;
	int answer = 0;
	for (int i = 0; i < str.length(); i++)
	{
		if (str.at(i) == '(')
		{
			s.push('(');
		}
		else
		{
			if (s.empty())
			{
				answer++;
			}
			else
			{
				s.pop();
			}
		}
	}
	answer += s.size();
	std:: cout << answer << "\n";
	return 0;
}