https://www.acmicpc.net/problem/2225

 

2225번: 합분해

첫째 줄에 답을 1,000,000,000으로 나눈 나머지를 출력한다.

www.acmicpc.net

 

풀이

이전 항으로 다음 항의 값을 구할 수 있는 다이나믹 프로그래밍 문제이다.

 

문제의 상황을 수식으로 나타내보면 다음과 같다.

 

 

이렇게 되는 경우의 수를 dp[k][N]이라고 할 떄, 위에 수식은 아래처럼 다시 바꿔서 표현할 수 있다.

 

 

L이라는 값을 미리 더해서 (0 ~ n-L)까지 k-1개의 숫자를 더해서 N-L을 만드는 경우의 수를 구하는 상황으로 바꿀 수 있고, k-1개로 묶인 부분이 dp[k-1][N-L]이 된다.

 

그러면 dp[k][N] = sum( dp[k-1][N-L] ) (0 <= L <= N)이라는 점화식을 만들어낼 수 있는데, 변수가 3개이므로 3중 반복문을 사용해서 구현할 수 있다.

 

 

dp[k][N]의 식을 조금 살펴보면,

k = 1일 때 즉, 숫자 하나로 N을 만드는 경우의 수는 N이 무엇이든 항상 N 하나이므로 dp[1][N] = 1이다. 그리고 이게 초기 값이 된다.

n =0일 때는 숫자가 몇 개든지 0만 더해지는 경우밖에 없다. 따라서 dp[k][0] = 1이다. 

ex) dp[4][0] = 1 (0 + 0 + 0 +0)

그리고 이것들이 dp의 초기 값이 된다.

 

머리로만 하면 이해가 잘 안될 수도 있어서 직접 해보는 게 좋다. 내가 그랬다ㅋ_ㅋ

 

ex) n = 6, k = 4일 때 반복문 결과

dp[k][n] n = 0 1 2 3 4 5 6
k =1 1 1 1 1 1 1 1
2 1 1+1=2 1+1+1=3 4 5 6 7
3 1 1+2=3 1+2+3=6 10 15 21 28
4 1 1+3=4 1+3+6=10 20 35 56 56+28=84

  

계산하다 보면 dp[k][n] = dp[k][n-1] + dp[k-1][n] 이라는 특이한 구조를 발견할 수 있다. 이 점화식을 사용하면 변수가 두 개라서 이중반복문으로 좀 더 빠르게 구현이 가능하다!

 

삼중반복문이 처음엔 머리에 잘 안들어왔었는데, 직접 해보는게 이해에 도움이 많이 됐던 문제다.

 

코드1 (삼중 반복문)

fun main(){
    val input = readLine()!!.trim().split(" ")
    var n = input[0].toInt()
    var k = input[1].toInt()
    val dp = Array(k+1){IntArray(n+1)}

    for (i in 0..n) dp[1][i] = 1

    for (i in 2..k){		// 행: 더하는 숫자 개수
        for (j in 0..n){	// 열: 더해서 나와야 하는 값
            for (m in 0..j){
                dp[i][j] += dp[i-1][j - m] // 직전 행의 처음~현재 열까지 더한다.
                dp[i][j] %= 1_000_000_000
            }
        }
    }
    
    println(dp[k][n])
}

 

코드2 (재귀)

lateinit var dp: Array<IntArray>

fun main() {
    val input = readLine()!!.trim().split(" ")
    var n = input[0].toInt()
    var k = input[1].toInt()
    dp = Array(k+1){IntArray(n+1)}

    for (i in 0..n){
        dp[1][i] = 1	// 더하는 숫자가 하나일 때
    }

    println(findAnswer(n, k))
}

private fun findAnswer(n: Int, k:Int): Int {
    if (dp[k][n] != 0) return dp[k][n]

    for (i in 0..n){
        dp[k][n] += findAnswer(n-i, k-1)
        dp[k][n] %= 1_000_000_000
    }

    return dp[k][n]
}

 

코드3 (이중 반복문)

fun main(){
    val input = readLine()!!.trim().split(" ")
    var n = input[0].toInt()
    var k = input[1].toInt()
    val dp = Array(k+1){IntArray(n+1)}

    for (i in 1..k) dp[i][0] = 1 // 합한 값이 0인 경우는 항상 한 가지이다
    for (i in 1..n) dp[0][i] = 0 // dp[1][i] = 1로 세팅하고, 반복문을 2..k으로 시작하는 거랑 같음 

    for (i in 1..k){
        for (j in 1..n){
            dp[i][j] = dp[i][j-1] + dp[i-1][j]
            dp[i][j] %= 1_000_000_000
        }
    }

    println(dp[k][n])
}

https://www.acmicpc.net/problem/2075

 

2075번: N번째 큰 수

첫째 줄에 N(1 ≤ N ≤ 1,500)이 주어진다. 다음 N개의 줄에는 각 줄마다 N개의 수가 주어진다. 표에 적힌 수는 -10억보다 크거나 같고, 10억보다 작거나 같은 정수이다.

www.acmicpc.net

풀이

간단한 규칙이 있는 이차원 배열 수들 중 n번째로 큰 수를 찾는 문제였다.

 

공간 복잡도: 1500 *1500 * 4 byte = 9MB

시간 복잡도: 2,250,000 * log(2,250,000) = 2,250,000 * 21. ~ = 약 47,250,000

처음엔 복잡도가 충분해보였고 혹시 공간 복잡도에 문제가 있을까 싶었지만, 우선 일차원 배열을 정렬해서 답을 구해봤다. 그리고 예상외로 시간초과 판정을 받았다.

왜 시간초과가 발생했는지 찾아봤는데, 자바의 Array와 Collections의 정렬 방식이 다르기 때문이었다.
정렬의 실제 시간 복잡도는 C * 시간복잡도 + a로 알고리즘에 따라 C값이 달라지는데, Array는 퀵정렬을, Collections은 Tim 정렬이라는 걸 사용한다고 한다. 그 이유는, Array는 데이터들이 메모리에 연속적으로 존재하는 반면, list는 산발적인 연결리스트 방식으로도 존재해서 참조 지역성에 차이가 있기 때문이다. 그래서 리스트를 사용했을 때 시간 초과가 났고, IntArray로 풀어봤더니 정말로 됐다ㅋㅋ 

(자세한 내용은 https://sabarada.tistory.com/138 여기를 참고했습니다ㅋ.ㅋ)

 

 

다시 문제로 돌아와서, 고민하다가 풀이를 찾아보니 heap 또는 우선순위 큐를 사용하는 문제였다. 문제에서 주어진 배열의 크기 조건에 관계 없이 heap의 특징을 사용해서 쉽게 푸는 방법문제 조건을 함께 사용하는 방법이 있었고, 후자의 방법이 조금 더 빨랐다. 

참고) https://jihogrammer.tistory.com/25

 

 

방법1)

아이디어는 간단하다. 연산을 끝마쳤을 때 heap에 가장 큰 수 n개가 들어있게 만든다면, 그 중 최소 값이 n번째로 큰 수가 된다. 그러기 위해서, 최소heap에 수를 하나씩 넣는다. 그리고 크기를 n으로 유지하기 위해 최소 값을 삭제하고 새로운 숫자를 삽입하면, 연산할 때마다 최소 값이 하나씩 사라지므로 연산이 끝났을 때의 n개의 숫자들이 가장 큰 수의 집합이 된다. 

 

방법2)

두 번째 방법은 방법1에서 heap의 연산(logn) 횟수를 절약하는 것이다. 방법1에서는 연산을 할 때마다 heap에서 값을 빼고 넣는데, 사실 heap의 목적은 최소 값을 버림으로써 지금까지의 숫자들 중 최대 값 n개를 유지하는 것이다. 따라서, heap의 최소 값보다 작은 수는 연산할 필요가 없으므로 생략한다.

 

방법3)

세 번째는 heap과 더불어 문제의 조건도 함께 사용하는 방법이다. 다시 한번 말하지만, 핵심은 연산할 때마다가 아닌, 연산이 모두 끝났을 때 가장 큰 숫자 n개를 갖고 있는 것이다!

 

아이디어) 최대 heap을 사용한다. 처음에 heap이 전체의 최대 값을 갖고 있고, n-1번의 연산 동안 최대 값을 그 다음의 최대 값으로 교체할 수 있다면, 그때 최대heap의 값이 답이 된다.

 

이해하면 간단할 수도 있는데, 다이나믹 프로그래밍적인 느낌도 들고 처음부터 떠올리기엔 쉽지 않은 것 같다ㅋㅋ

 

구현 방법)

1. 이차원 배열에 숫자를 입력 받는다.

2. 마지막 행의 수들을 최대heap에 넣는다.

3. 그러면 heap은 처음에 가장 큰 수를 포함해서 각 열의 최대 값을 갖게 된다.

4. heap의 최대 값을 삭제한다.

5. 삭제한 값을 numbers[row][col]라고 할 때, heap에 numbers[row-1][col]을 삽입한다. 그럼 항상 모든 열의 최대 값을 heap이 갖게 되는데, 전체 수의 최대 값은 항상 모든 열의 최대 값 중 하나이다.

6. 따라서 이 연산을 n-1번 반복했을 때, 힙의 최대 값이 n번째로 큰 수가 된다.

 

구현을 위해선 이차원 배열에 값을 저장해놓고, 힙에 배열의 좌표를 함께 저장해야 한다. 

 

코드1

import java.util.*

fun main() {
    val n = readLine()!!.toInt()
    val pq = PriorityQueue<Int>() // 최소힙

    // 힙의 사이즈를 n으로 유지한다
    pq.addAll(readLine()!!.trim().split(" ").map{it.toInt()}) // 힙에 첫번째 줄 삽입
    repeat(n-1){
        val list = readLine()!!.trim().split(" ").map{ it.toInt() }
        list.forEach { number ->
            pq.offer(number)
            pq.poll()
        }
    }

    // 가장 큰 n개의 수만 남아있다
    println(pq.poll())
}

 

코드2

import java.util.*

fun main() {
    val n = readLine()!!.toInt()
    val pq = PriorityQueue<Int>(n) // 최소힙

    pq.addAll(readLine()!!.trim().split(" ").map{it.toInt()}) // 첫번째 줄 삽입
    
    repeat(n-1){
        val list = readLine()!!.trim().split(" ").map{ it.toInt() }
        list.forEach { number ->
            // 최대 값 n개를 유지한다 -> 힙의 최소 값보다 작은 수는 넣을 필요가 없다
            // 힙 연산 횟수를 줄이기 위해 힙의 최소 값보다 작은 수는 생략
            if (pq.peek() < number){	  
                pq.poll()
                pq.offer(number)
            }
        }
    }

    // 가장 큰 n개의 수만 남아있다
    println(pq.poll())
}

 

코드3

// 시간을 더 단축해보고 싶어서 StringTokenizer 사용했습니다.

import java.io.BufferedReader
import java.io.InputStreamReader
import java.util.*

data class Number(val row: Int, val column: Int, val number: Int)

fun main() = with(BufferedReader(InputStreamReader(System.`in`))) {
    val n = readLine().toInt()
    val numbers = Array(n){ mutableListOf<Number>() }

    // 이차원 배열에 좌표와 함께 저장
    repeat(n){ i ->
        val st = StringTokenizer(readLine())
        for(j in 0 until n){
            val number = st.nextToken().toInt()
            numbers[i].add(Number(i, j, number))
        }
    }

    // 최대heap
    val pq = PriorityQueue<Number>(compareByDescending{it.number}).apply{
        addAll(numbers[n-1])    // 가장 큰 수가 들어있는 마지막 라인 추가
    }
    
    // 가장 큰 수를 차례대로 n-1번 꺼낸다
    repeat(n-1){
        val maxNumber = pq.poll()
        pq.add(numbers[maxNumber.row-1][maxNumber.column]) // 그 다음으로 가장 큰 수 n개 유지
    }

    println(pq.peek().number)
}

 

https://www.acmicpc.net/problem/19598

 

19598번: 최소 회의실 개수

서준이는 아빠로부터 N개의 회의를 모두 진행할 수 있는 최소 회의실 개수를 구하라는 미션을 받았다. 각 회의는 시작 시간과 끝나는 시간이 주어지고 한 회의실에서 동시에 두 개 이상의 회의

www.acmicpc.net

 

풀이

가장 먼저 시작되는 회의에 회의실을 배정한다. 그리고 현재 진행중인 회의들 중 가장 먼저 끝나는 회의와 다음에 시작할 회의의 시작 시간을 비교해서, 진행중인 회의의 끝나는 시간이 더 빠르다면 그 강의실에 새로 시작할 회의를 그대로 넣어준다. 반대로 새로 시작하는 시간이 더 빠르다면 새 회의실을 사용한다. 이렇게 하나씩 비교를 끝마치면, 남아있는 회의의 개수가 지금껏 사용한 회의실의 개수가 된다. 기존의 회의실만 삭제하는 경우는 없으며, 1:1로 교체하거나 추가로 생성만 하기 때문이다.

 

자료구조)

비교를 할 때 남아있는 회의들은 시작 시간순으로 접근하므로, 시작시간의 오름차순으로 정렬을 하고,

진행중인 회의들 중에서는 가장 빨리 끝나는 회의 하나만 꺼내서 비교하므로 끝나는 시간을 기준으로 우선순위 큐(최소힙)를 사용하면 편하게 구현할 수 있다.

 

 

우선순위 큐가 회의실이고, 회의=회의실로 생각해서 교체, 추가한다는 발상이 정말 인상적이었다.

 

코드

import java.io.BufferedReader
import java.io.InputStreamReader
import java.util.*

data class Meeting(val start: Int, val end: Int)

fun main() = with(BufferedReader(InputStreamReader(System.`in`))) {
    val n = readLine().toInt()
    val meetings = mutableListOf<Meeting>()
    val pq = PriorityQueue<Meeting>(compareBy{ it.end }) // 현재 진행중인 회의 저장(사용중인 회의실)

    repeat(n){
        val st = StringTokenizer(readLine())
        val meeting = Meeting(st.nextToken().toInt(), st.nextToken().toInt())
        meetings.add(meeting)
    }
    meetings.sortWith(compareBy{ it.start }) // 일찍 시작하는 순서대로 정렬

    pq.add(meetings[0])
    for (i in 1 until n){
        val prevMeeting = pq.peek()   // 진행중인 회의 중 가장 먼저 끝나는 회의
        val nextMeeting = meetings[i] // 다가올 회의 중 가장 먼저 시작하는 회의
        // 기존 회의가 먼저 끝나면 그 회의실에서 다음 회의를 진행할 수 있으므로, 기존 회의를 삭제한다
        if (prevMeeting.end <= nextMeeting.start){
            pq.poll()
        }

        pq.offer(nextMeeting)
    }

    // 회의를 1:1로 교체하거나 회의실이 부족하면 추가 회의실을 사용했기 때문에
    // 남아있는 회의의 개수 = 지금껏 사용한 회의실 개수
    println(pq.size)
}

https://www.acmicpc.net/problem/2579

 

2579번: 계단 오르기

계단 오르기 게임은 계단 아래 시작점부터 계단 꼭대기에 위치한 도착점까지 가는 게임이다. <그림 1>과 같이 각각의 계단에는 일정한 점수가 쓰여 있는데 계단을 밟으면 그 계단에 쓰여 있는 점

www.acmicpc.net

 

코드

포도주 시식 문제(https://www.acmicpc.net/problem/2156)와 정말 비슷하다.

포도주 시식 포스팅: (https://best-human-developer.tistory.com/19)

 

다른 점이 하나 있는데, 포도주 문제에서는 i번째 포도주를 선택하지 않아도 되지만 계단 오르기에서는 i번째 계단을 반드시 밟아야 한다. i번째 계단을 밟지 않는 경우를 포함시키면 3칸 이상을 건너뛰는 상황이 발생해서, 한 번에 한 계단 또는 두 계단씩만 오를 수 있다는 조건1을 만족시키지 못하기 때문이다.

 

따라서,

함수) dp[i]: i번째 계단까지만 존재할 때, 조건1,2,3을 만족하는 최대 점수

점화식) i번째 계단 + i-1번째 계단을 밟는 경우와 i번째 계단 + i-2번째 계단을 밟는 경우 중 최대 값

-> dp[i] = max( stair[i] + stair[i-1] + dp[i-3] , stair[i] + dp[i-2] ) 

초기값) dp[0] = 1, dp[1] = stair[1], dp[2] = stair[1] + stair[2]

 

와 같이 반복문을 구성하면, 초기 값부터 조건을 만족하면서 모든 경우의 수를 포함하기 때문에 dp[3] ~ dp[n]까지 답을 구할 수 있다.

 

정답) dp[n]

 

풀이

import kotlin.math.max

fun main() {
    val n = readLine()!!.toInt()
    val stairs = IntArray(n+2)	// n+2: 초기 값 설정할 때 인덱스 에러를 막는다 
    val dp = IntArray(n+2)

    repeat(n) { i ->
        stairs[i+1] = readLine()!!.toInt()
    }

    dp[1] = stairs[1]
    dp[2] = stairs[1] + stairs[2]
    for (i in 3..n){
        // 직전 계단+현재 계단, 두 칸 전 계단+현재 계단 중 최대 값
        // 현재 계단을 밟지 않는 경우도 포함시키면, 세 칸 이상을 뛰어넘는 경우가 발생한다
        dp[i] = max(stairs[i] + stairs[i-1] + dp[i-3], stairs[i] + dp[i-2])
    }

    println(dp[n])
}

+ Recent posts