PS/백준

[BOJ] 백준 20922번: 겹치는 건 싫어 (Kotlin)

제이온 (Jayon) 2023. 1. 23. 17:38

 

문제

홍대병에 걸린 도현이는 겹치는 것을 매우 싫어한다. 특히 수열에서 같은 원소가 여러 개 들어 있는 수열을 싫어한다. 도현이를 위해 같은 원소가 K개 이하로 들어 있는 최장 연속 부분 수열의 길이를 구하려고 한다.

 

 100000 이하의 양의 정수로 이루어진 길이가 N인 수열이 주어진다.  이 수열에서 같은 정수를 K개 이하로 포함한 최장 연속 부분 수열의 길이를 구하는 프로그램을 작성해보자.

 

입력

첫째 줄에 정수 N (1 <= N <= 200000)과 K (1 <= K <= 100)이 주어진다.

둘째 줄에는 a1, a2, ..., an이 주어진다. (1 <= ai <= 100000)

 

출력

조건을 만족하는 최장 연속 부분 수열의 길이를 출력한다.

 

풀이

투 포인터로 해결할 수 있는 문제였습니다.

전형적인 부분 합을 이용한 투 포인터 코드와 유사한데, 부분 합 대신 중복된 원소의 개수로 치환하여 문제를 풀면 됩니다. 이때 특정 원소의 중복 횟수를 체크하기 위해서 Hash Map을 사용했습니다.

 

소스 코드

import java.io.BufferedReader
import java.io.BufferedWriter
import java.io.InputStreamReader
import java.io.OutputStreamWriter

fun main() {
    val reader = BufferedReader(InputStreamReader(System.`in`))
    val writer = BufferedWriter(OutputStreamWriter(System.`out`))

    val inputs = reader.readLine().split(" ").map { it.toInt() }
    val n = inputs[0]
    val k = inputs[1]
    val arr = reader.readLine().split(" ").map { it.toInt() }

    var start = 0
    var end = 0
    var count = 0
    var answer = Int.MIN_VALUE

    val map = mutableMapOf<Int, Int>()
    while (start < n) {
        while (end < n) {
            val value = map.getOrDefault(arr[end], 0)
            if (value >= k) {
                break
            }
            map[arr[end++]] = value + 1
            count++
        }
        answer = answer.coerceAtLeast(count--)
        map[arr[start]] = map[arr[start]]!! - 1
        start++
    }

    writer.write("$answer\n")
    writer.flush()
    writer.close()
    reader.close()
}