알고리즘 문제/Leetcode

[Leetcode] 1658. Minimum Operations to Reduce X to Zero

햄과함께 2022. 6. 11. 17:36
320x100

문제 : https://leetcode.com/problems/minimum-operations-to-reduce-x-to-zero/


정수 배열 nums와 정수 x가 입력으로 주어진다.

한 번의 작업으로 배열 nums에서 가장 좌측 혹은 가장 우측에 있는 요소를 제거 하고 x에서 제거한 요소의 값을 뺄 수 있다.

최소 몇 번의 작업으로 x를 정확히 0으로 만들 수 있는지 구하라. 만일 불가능하다면 -1을 반환하라.


부분합과 투포인터로 구할 수 있다.

[그림1]

부분합을 만들어 subsum 배열에 세팅한다.

첫 번째 예제 [1, 1, 4, 2, 3], x=5 로 표를 세팅해보면 [그림1]과 같다.

정답이 되는 제거되는 요소는 가장 뒤 요소 2, 3이다.

이를 반대로 생각하면 제거되지 않는 요소들의 합은 nums 전체 배열 합 11에 x, 5를 뺀 6이 된다.

투 포인터로 제거되지 않는 요소들의 범위를 구하고 이 범위의 합을 subSum 배열로 O(1)만에 구한다.

구한 합이 6과 같다면 해당 범위에 속하지 않은 범위의 요소들의 개수들 중 최소값이 정답이 된다.

 

만일 범위의 합이 6과 같다면 정답변수와 비교하여 정답변수보다 작다면 정답변수를 갱신한다.

6보다 작다면 범위를 늘려야 하므로 right를 1 증가한다.

6보다 크다면 범위를 줄여야 하므로 left를 1 증가한다.

left는 항상 right 보다 작거나 같아야 한다.

 

예제 1로 예를 들어보자.

right = 0

right가 0, left가 0인 경우 범위의 합은 1로 6보다 작다. 따라서 범위 확장을 위해 right를 1 증가시킨다.

right = 1

right가 1, left가 0인 경우 범위의 합은 2로 6보다 작다. 따라서 right를 1 증가시킨다.

right = 2

right가 2, left가 0인 경우 범위의 합은 6으로 6과 같다. 따라서 정답일 가능성이 될 수 있기때문에 정답변수를 2(해당 범위에 속하지 않은 요소들의 수)로 갱신한다.

갱신 후 다음 범위 탐색을 위해 right를 1 증가시킨다.

right =3

right가 3, left가 0인 경우 범위의 합은 8로 6보다 크다. 범위를 줄이기 위해 left를 1 증가시킨다.

 

right가 3, left가 1인 경우 범위의 합은 7로 6보다 크다. 범위를 줄이기 위해 left를 1 증가시킨다.

 

right가 3, left가 2인 경우 범위의 합은 6으로 6과 같다. 따라서 정답일 가능성이 될 수 있기때문에 정답변수를 3(해당 범위에 속하지 않은 요소들의 수)과 비교한다. 정답변수(2)가 더 작기 때문에 갱신하지 않고 다음 범위 탐색을 위해 right를 1 증가시킨다.

right = 4

right가 4, left가 2인 경우 범위의 합은 9로 6보다 크다. 범위를 줄이기 위해 left를 1 증가시킨다.

 

right가 4, left가 3인 경우 범위의 합은 5로 6보다 작다. 범위를 늘리기 위해 right를 1 증가시킨다.

 

right가 배열 범위를 벗어나기 때문에 탐색을 종료한다.

 

 

공간, 시간복잡도는 O(N).


소스코드 : https://github.com/fpdjsns/Algorithm/blob/master/leetcode/medium/1658.%20Minimum%20Operations%20to%20Reduce%20X%20to%20Zero.cpp


문제 괜찮구먼. 부분합 배열 안만들어도 최신 부분합 값만 가지고 있으면 되므로 공간복잡도 O(1)로 줄일 수 있다.

320x100