[C#]프로그래머스-풍선 터트리기

2021. 4. 15. 01:27[C#] 코딩테스트 문제풀이

반응형

프로그래머스 문제

 

문제 설명

일렬로 나열된 n개의 풍선이 있습니다. 모든 풍선에는 서로 다른 숫자가 써져 있습니다. 당신은 다음 과정을 반복하면서 풍선들을 단 1개만 남을 때까지 계속 터트리려고 합니다.

  1. 임의의 인접한 두 풍선을 고른 뒤, 두 풍선 중 하나를 터트립니다.
  2. 터진 풍선으로 인해 풍선들 사이에 빈 공간이 생겼다면, 빈 공간이 없도록 풍선들을 중앙으로 밀착시킵니다.

여기서 조건이 있습니다. 인접한 두 풍선 중에서 번호가 더 작은 풍선을 터트리는 행위는 최대 1번만 할 수 있습니다. 즉, 어떤 시점에서 인접한 두 풍선 중 번호가 더 작은 풍선을 터트렸다면, 그 이후에는 인접한 두 풍선을 고른 뒤 번호가 더 큰 풍선만을 터트릴 수 있습니다.

당신은 어떤 풍선이 최후까지 남을 수 있는지 알아보고 싶습니다. 위에 서술된 조건대로 풍선을 터트리다 보면, 어떤 풍선은 최후까지 남을 수도 있지만, 어떤 풍선은 무슨 수를 쓰더라도 마지막까지 남기는 것이 불가능할 수도 있습니다.

일렬로 나열된 풍선들의 번호가 담긴 배열 a가 주어집니다. 위에 서술된 규칙대로 풍선들을 1개만 남을 때까지 터트렸을 때 최후까지 남기는 것이 가능한 풍선들의 개수를 return 하도록 solution 함수를 완성해주세요.

 

제한 사항

  • a의 길이는 1 이상 1,000,000 이하입니다.
    • a[i]는 i+1 번째 풍선에 써진 숫자를 의미합니다.
    • a의 모든 수는 -1,000,000,000 이상 1,000,000,000 이하인 정수입니다.
    • a의 모든 수는 서로 다릅니다.

 

내 풀이

using System;
using System.Collections.Generic;
public class Solution {
    public int solution(int[] a) {
        int answer = a.Length;
        int leftmin = a[0]; //left 초기값
        int rightmin = a[a.Length-1]; //right 초기값
        List<bool> temp = new List<bool>(); 
        for(int i=0; i<a.Length; i++)
        {
            temp.Add(false);
        }
        
        for(int i=0; i<a.Length; i++)
        {
            if(leftmin<a[i])
            {
                temp[i]=true;
            }
            else
            {
                temp[i] =false;
                leftmin = a[i];
            }
        }
        for(int i=a.Length-2; i>=0; i--)
        {
            if(rightmin > a[i])
            {
                rightmin = a[i];
            }
            else if(temp[i]==true)
            {
                answer--;
            }
        }
        return answer;
    }
}

 

풀이 : 규칙은 쉽게 도출해냈다.

 

1. 자신 기준 왼쪽 풍선이 없는 첫번째 풍선과 자신 기준 오른쪽 풍선이 없는 마지막 풍선은 return.

 

2. 첫번째와 마지막에 위치하지 않은 풍선들은 자신 기준 왼쪽 풍선들의 가장 작은 값 leftmin과 자신 기준 오른쪽 풍선들의 가장 작은

 

rightmin 도출.

 

3. if leftmin과 rightmin 모두 타겟 풍선의 숫자보다 작다면 return 불가능한 타겟.

 

이 세가지 규칙을 준수하면 예외없이 정답을 도출 해낼 수 있다. 하지만 이 문제의 중요한 포인트는 제한사항에 있었다.

 

조건 [ a의 모든 수는 -1,000,000,000 이상 1,000,000,000 이하인 정수입니다. ] 때문에

 

이중 반복문을 사용해 수행시간 n^2으로 푼다면 간단하지만 런타임 에러 문제에 봉착.

 

때문에 수행 시간을 2n으로 줄이기 위해, 배열 크기의 불형 리스트 temp를 두고

 

왼쪽 풍선부터 leftmin을 구해주는 반복문 하나,오른쪽 풍선부터  rightmin을 구해주는 반복문을 하나를 두고

 

자신 기준 leftmin과 rightmin이 모두 자신보다 작으면 temp의 bool값이 true,

 

temp[i]가 true라면 answer -1 해주어 해결.

 

느낀점:런타임 에러에 봉착한 후 수행시간을 줄이는 것이 아직 익숙하지 못한 느낌을 받았다.

 

실제 코드를 짤때에도 코딩테스트 문제를 풀 때에도 항상 수행시간을 최대한 줄이는 습관을 생활화 해봐야 할 것 같다.

반응형