본문 바로가기

Developer/Algorithm

[프로그래머스] 구현 - 풍선터트리기 (Javascript)

 [ 문제 설명 ] 

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

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

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

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

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

 

 [ 제한 사항 ] 

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

 

 

 [ 문제 풀이 ] 

//특정 선택 값의 앞 뒤의 값들이 특정 값보다 크거나 없으면 성공 -> 특정 값이 앞뒤의 값들보다 크면 실패
방법 1) for 문을 돌리고 왼쪽 오른쪽 최소값이 측정값보다 크거나 없으면 숫자 증가 => O(n^2) 시간초과...

방법 2)
        1) Math.min과 slice 를 사용하지 않고 for문을 돌릴 때마다 왼쪽 오른쪽의 최솟값 저장
        2) now 값이 왼쪽 오른쪽의 값보다 크지 않다면 각 arr에서 값 저장 
        3) arr의 길이 값 더하기 끝에 2개 값이 정답

 

 

 [ 문제 코드 ] 

///방법 1 => 효율성 실패
function solution(a) {
    let answer = 0;
    for(let i=0; i<a.length; i++){
        let now = a[i];
        let left = Math.min(...a.slice(0,i));
        let right = Math.min(...a.slice(i+1,a.length));
        if(!left || !right || now<left || now<right){
            answer++
        }
    }
    return answer
}

///방법 2
function solution(a) {
  var left_arr = []
  var right_arr = []
  var left_min = a[0]
  var right_min = a[a.length - 1]
  for (var i = 1; i < a.length - 1; i++) {
    if (left_min > a[i]) {
      left_min = a[i]
      left_arr.push(a[i]) /// 왼쪽에서 최솟값보다 작은 값들을 넣어줌 (성공케이스)
    }
    if (right_min > a[a.length - i - 1]) {
      right_min = a[a.length - i - 1]
      right_arr.push(a[a.length - i - 1]) ///오른쪽에서 최솟값보다 작은 값들을 넣어줌 (성공케이스) 
    }
  }
  return [...new Set([...left_arr, ...right_arr])].length + 2; ///Set 함수의 특성(중복값 제거)을 사용한 후, 갯수 + 2(앞,뒤)  
}

 

 [ 셀프 피드백 ] 

Math.min과 slice를 사용하여 푸는 방법에서 효율성 실패로 다소 시간이 오래 걸렸다.

많이 연습해서 미리 최솟값을 저장하는 부분을 빠르게 구현할 수 있도록 해야겠다.