알고리즘 문제 풀이

[백준 24552] 올바른 괄호

다빈치코딩 2024. 11. 27. 19:55
반응형

올바른 괄호(24552)

문제 출처 : https://www.acmicpc.net/problem/24552

 

이 문제를 처음 보았을 때 스택을 이용하여 문제를 해결한다고 생각했습니다. 보통 괄호 문제는 스택에 넣어가며 문제를 해결하기가 쉽기 때문 입니다. 아직 괄호 문제를 풀어본 경험이 없다면 먼저 아래 문제부터 해결해 보시기 바랍니다.

https://davincicoding.tistory.com/189

 

[백준 9012] 괄호

괄호(9012)문제 출처 : https://www.acmicpc.net/problem/9012 이 문제는 다빈치코딩 알고리즘에서 이미 스택으로 풀어본 문제 입니다. 이전 스택 풀이는 아래 링크를 확인 바랍니다.https://wikidocs.net/215110 06.

davincicoding.co.kr

 

위 링크를 통해 수학적으로 해결하는 방식을 이해해야 이 문제 풀이를 보다 쉽게 이해할 수 있습니다.

문제 이해하기

이 문제는 올바른 괄호가 구성되어 있는지 확인하는 문제가 아닙니다. 문제 자체에는 잘못된 괄호가 주어지고, 여기서 하나의 괄호를 빼서 올바른 괄호를 만드는 것입니다. 이 때 올바른 괄호를 만들기 위해 몇 개의 괄호 문자열을 제거할 수 있는지 경우의 수를 묻는 문제 입니다.

어느 괄호를 지워야 하는가?

( ) ( ( ) ) )

위와 같이 문제의 예제처럼 괄호가 주어졌을 때 어느 괄호를 뺄 수 있을지 생각해 보겠습니다.

( ) ( ( ) ) )

열리는 괄호는 파란색, 닫는 괄호는 빨간색 입니다. 괄호의 개수는 파란색은 3개, 빨간색은 4개 입니다. 즉 빨간 괄호를 1개 줄일 때 파란색과 빨간색의 숫자가 같아지게 됩니다. 이 예제에서는 빨간 괄호의 개수 4가 답이 됩니다.

지울 수 없는 괄호는?

하지만 이렇게 괄호의 개수로만 문제를 파악해서는 안됩니다. 왜냐하면 괄호는 열리고, 닫혀야 하는데 닫히는 괄호가 먼저 나오면 안되기 때문 입니다.

( ) ( ( )

이번에는 파란 괄호가 3개이고, 빨간 괄호가 2개 입니다. 위 예제처럼 단순히 파란 괄호의 개수가 더 많기 때문에 파란 괄호의 개수인 3이라고 답을 하면 틀리게 됩니다. 왜냐하면 첫 번째 파란색 괄호를 빼는 순간 올바른 괄호가 될 수 없기 때문 입니다.

) ( ( )

첫 번째 파란색 괄호를 빼면 위와 같은 모양이 되면서 올바른 괄호가 되지 않습니다. 괄호가 열리고 닫히는 것을 파악해서 더 이상 뺄 수 없는 괄호는 건드리지 않아야 합니다. 즉 괄호의 합계가 0이 되는 순간이 온다면 그 이전의 괄호에서는 뺄 수 있는 괄호가 없다는 뜻입니다.

합계가 0보다 작아진다면?

앞서 9012 문제에서는 괄호의 합계가 0보다 작아지면 올바른 괄호를 만들 수 없다고 바로 판단하였습니다. 하지만 이 문제에서는 그런 걱정은 할 필요가 없습니다. 왜냐하면 이 문제에서는 답이 1 이상이라고 정의를 했기 때문 입니다. 즉 합계가 0 이하라면 우리가 지워야 할 대상이라는 뜻입니다.

( ( ) ) ) ( )

위 예제에서 빨간색 괄호 3번째가 되었을 때 괄호의 합계가 0보다 작아지게 됩니다. 그럼 앞선 빨간색 괄호의 개수 3개가 답이 됩니다.

코드 작성하기

그럼 코드를 작성해보도록 하겠습니다.

입력 받고, 답 출력하기

S = input()

print(solve(S))

괄호 문자열 S를 입력 받습니다. 그리고 solve 함수를 통해 원하는 결과를 얻어 출력 합니다.

solve 함수

def solve(arr):
    left_cnt = 0
    right_cnt = 0
    total = 0

    for s in arr:
        if s == '(':
            left_cnt += 1
            total += 1
        elif s == ')':
            right_cnt += 1
            total -= 1
        
        if total < 0:
            return right_cnt
        
        if total == 0:
            left_cnt = 0

    return left_cnt

9012번과 유사합니다. 다른점은 ‘(’를 만났을 때 left_cnt를 늘려줍니다. 그리고 ‘)’를 만나면 right_cnt를 늘려줍니다.

total이 0보다 작아진다는 것은 괄호가 이미 올바르지 않다는 뜻입니다. 이 때에는 right_cnt가 답이 됩니다. 위에서 합계가 0보다 작아지는 경우가 이 경우 입니다.

total 값이 0인경우 left_cnt 값을 0으로 초기화 합니다. total이 0이되면 더 이상 그 괄호 문자열에서는 뺄 괄호가 없기 때문에 0으로 초기화를 시켜주어야 합니다.

마지막으로 left_cnt 값을 출력해주면 답이 됩니다. right_cnt값을 답으로 출력하지 않는 이유는 right_cnt 값은 total 값이 0보다 작은 경우에 이미 쓰였습니다. 최종적으로 모든 반복문을 다 돌았다면 total 값이 0보다 크다는 뜻이고, 그것은 left_cnt 값만 의미가 있다는 뜻입니다.

전체 코드

전체 코드를 알아보겠습니다.

S = input()

def solve(arr):
    left_cnt = 0
    right_cnt = 0
    total = 0

    for s in arr:
        if s == '(':
            left_cnt += 1
            total += 1
        elif s == ')':
            right_cnt += 1
            total -= 1
        
        if total < 0:
            return right_cnt
        
        if total == 0:
            left_cnt = 0

    return left_cnt

print(solve(S))
반응형