문제 링크
https://school.programmers.co.kr/learn/courses/30/lessons/68646
프로그래머스
코드 중심의 개발자 채용. 스택 기반의 포지션 매칭. 프로그래머스의 개발자 맞춤형 프로필을 등록하고, 나와 기술 궁합이 잘 맞는 기업들을 매칭 받으세요.
programmers.co.kr
코딩테스트 연습 > 월간 코드 챌린지 시즌1 > 풍선 터트리기
문제 설명
일렬로 나열된 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의 모든 수는 서로 다릅니다.
입출력 예
a | result |
[9,-1,-5] | 3 |
[-16,27,65,-2,58,-92,-71,-68,-61,-33] | 6 |
입출력 예 설명
입출력 예 #1
- 첫 번째 풍선(9가 써진 풍선)을 최후까지 남기는 방법은 다음과 같습니다.
- [9, -1, -5] 에서 -1, -5가 써진 풍선을 고른 뒤, -1이 써진 풍선(번호가 더 큰 것)을 터트립니다.
- [9, -5] 에서 9, -5가 써진 풍선을 고른 뒤, -5가 써진 풍선(번호가 더 작은 것)을 터트립니다.
- 두 번째 풍선(-1이 써진 풍선)을 최후까지 남기는 방법은 다음과 같습니다.
- [9, -1, -5] 에서 9, -1이 써진 풍선을 고른 뒤, 9가 써진 풍선(번호가 더 큰 것)을 터트립니다.
- [-1, -5] 에서 -1, -5가 써진 풍선을 고른 뒤, -5가 써진 풍선(번호가 더 작은 것)을 터트립니다.
- 세 번째 풍선(-5가 써진 풍선)을 최후까지 남기는 방법은 다음과 같습니다.
- [9, -1, -5] 에서 9, -1이 써진 풍선을 고른 뒤, 9가 써진 풍선(번호가 더 큰 것)을 터트립니다.
- [-1, -5] 에서 -1, -5가 써진 풍선을 고른 뒤, -1이 써진 풍선(번호가 더 큰 것)을 터트립니다.
- 3개의 풍선이 최후까지 남을 수 있으므로, 3을 return 해야 합니다.
입출력 예 #2
- 최후까지 남을 수 있는 풍선은 -16, -92, -71, -68, -61, -33이 써진 풍선으로 모두 6개입니다.
나의 코드
// i 마다 i 포함 왼쪽으로 최솟값을 left , i 포함 오른쪽으로 최솟값을 right에 담음
// a[i]가 left[i] 보다 크고 right[i] 보다도 크면 생존 불가능
class Solution {
public int solution(int[] a) {
int answer = 0;
int[] leftMin = new int[a.length];
int[] rightMin = new int[a.length];
leftMin[0] = a[0];
for(int i=1; i<a.length; i++) {
leftMin[i] = Math.min(leftMin[i-1], a[i]);
}
rightMin[a.length-1] = a[a.length-1];
for(int i=a.length-2; i>=0; i--) {
rightMin[i] = Math.min(rightMin[i+1], a[i]);
}
for(int i=0; i<a.length; i++) {
if(a[i] > leftMin[i] && a[i] > rightMin[i]) answer++;
}
return a.length - answer;
}
}
풀이
- 임의의 풍선을 두 개 선택하여 크기가 작은 풍선은 1회만 살아남을 수 있고 나머지 경우는 크기가 큰 풍선만 살아남을 수 있다. 따라서 풍선이 살아남기 위해서는 왼쪽 또는 오른쪽의 당겨지는 풍선이 크거나 1회만 작은 경우에만 살아남을 수 있다. (2회 이상 크기가 작은 풍선이 온다면 살아남을 수 없다)
- 제일 왼쪽 값을 살린 후에 계속해서 왼쪽 풍선과 비교하여 작은 풍선의 값을 살리는 경우인 leftMin을 만들고 제일 오른쪽 값을 살린 후에 계속해서 오른쪽 풍선과 비교하여 작은 풍선의 값을 살리는 경우인 rightMin을 만든다.
- 그 후에 해당 위치의 a와 비교하여 a[i]보다 leftMin[i]과 rightMin[i]가 작다면 왼쪽으로 접근해도 오른쪽으로 접근해도 양쪽 다 작은 풍선이어서 2번이나 작은 풍선이 존재하기 때문에 살아남을 수 없다.
- 따라서 해당 위치의 a를 카운팅하여 총 a의 개수에서 제거하면 살아남을 수 있는 풍선의 개수이므로 answer에 저장하고 반환한다.
'Programmers' 카테고리의 다른 글
[프로그래머스 Level.3] 경주로 건설 (2020 카카오 인턴십) (Java) (0) | 2023.03.03 |
---|---|
[프로그래머스 Level.3] 다단계 칫솔 판매 (2021 Dev-Matching: 웹 백엔드 개발자(상반기)) (Java) (2) | 2023.03.02 |
[프로그래머스 Level.3] 보석 쇼핑 (2020 카카오 인턴십) (Java) (0) | 2023.02.28 |
[프로그래머스 Level.3] 가장 긴 팰린드롬 (연습문제) (Java) (0) | 2023.02.27 |
[프로그래머스 Level.3] 불량 사용자 (2019 카카오 개발자 겨울 인턴십) (Java) (0) | 2023.02.23 |