알고리즘 문제 풀기

백준 16401: 과자 나눠주기 - javascript(이분 탐색)

Fo_rdang 2024. 3. 14. 14:45
반응형

문제 출처 

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

 

16401번: 과자 나눠주기

첫째 줄에 조카의 수 M (1 ≤ M ≤ 1,000,000), 과자의 수 N (1 ≤ N ≤ 1,000,000)이 주어진다. 둘째 줄에 과자 N개의 길이 L1, L2, ..., LN이 공백으로 구분되어 주어진다. 과자의 길이는 (1 ≤ L1, L2, ..., LN ≤ 1,

www.acmicpc.net

문제

명절이 되면, 홍익이 집에는 조카들이 놀러 온다. 떼를 쓰는 조카들을 달래기 위해 홍익이는 막대 과자를 하나씩 나눠준다.

조카들이 과자를 먹는 동안은 떼를 쓰지 않기 때문에, 홍익이는 조카들에게 최대한 긴 과자를 나눠주려고 한다.

그런데 나눠준 과자의 길이가 하나라도 다르면 조카끼리 싸움이 일어난다. 따라서 반드시 모든 조카에게 같은 길이의 막대 과자를 나눠주어야 한다.

M명의 조카가 있고 N개의 과자가 있을 때, 조카 1명에게 줄 수 있는 막대 과자의 최대 길이를 구하라.

단, 막대 과자는 길이와 상관없이 여러 조각으로 나눠질 수 있지만, 과자를 하나로 합칠 수는 없다. 단, 막대 과자의 길이는 양의 정수여야 한다.

입력

첫째 줄에 조카의 수 M (1 ≤ M ≤ 1,000,000), 과자의 수 N (1 ≤ N ≤ 1,000,000)이 주어진다.

둘째 줄에 과자 N개의 길이 L1, L2, ..., LN이 공백으로 구분되어 주어진다. 과자의 길이는 (1 ≤ L1, L2, ..., LN ≤ 1,000,000,000) 를 만족한다.

출력

첫째 줄에 조카 1명에게 줄 수 있는 막대 과자의 최대 길이를 출력한다.

단, 모든 조카에게 같은 길이의 막대과자를 나눠줄 수 없다면, 0을 출력한다.

예제 입력 1 복사

3 10
1 2 3 4 5 6 7 8 9 10

예제 출력 1 복사

8

예제 입력 2 복사

4 3
10 10 15

예제 출력 2 복사

7

 

후기 

살짝 눈물 날 뻔했다. 

이분탐색은 세번째로 풀어보는건데, silver2문제 19분 만에 풀었다. 

두번 틀리고,,, 고쳐서 맞았다. ! 

나도 할 수 있다. 나를 의심하는 것이 스스로에 대한 배신이다. 

문제 풀이 힌트 

- 이분 탐색을 활용하자. 

문제 풀이 코드 

const input = require('fs').readFileSync('/dev/stdin').toString().trim().split('\n'); 
const [M, N] = input[0].split(' ').map(v => +v); 
const sticks = input[1].split(' ').map(v => +v).sort((a,b) => a-b); 

let min = 0; 
let max = Math.max(...sticks); 
while(min <= max){
    const mid = Math.floor((min + max)/2); 
    let cnt = 0; 
    //mid 막대 길이를 동생들에게 나눠줄 수 있는지 확인하는 반복문 
    for(let i=N-1; i>=0; i--){
      let num = Math.floor(sticks[i]/mid); 
        cnt+= num; 
        if(cnt >= M){
             break; 
        }
    }
    if(cnt >= M){
        min = mid+1;
    }else{
        max = mid-1;
    }
}
console.log(max);

Only 문제 풀이 

 

const input = require('fs').readFileSync('/dev/stdin').toString().trim().split('\n'); 
const [M, N] = input[0].split(' ').map(v => +v); 
const sticks = input[1].split(' ').map(v => +v).sort((a,b) => a-b); 

let min = 0; 
let max = Math.max(...sticks); 
while(min <= max){
    const mid = Math.floor((min + max)/2); 
    let cnt = 0; 
    for(let i=N-1; i>=0; i--){
      let num = Math.floor(sticks[i]/mid); 
        cnt+= num; 
        if(cnt >= M){
             break; 
        }
    }
    if(cnt >= M){
        min = mid+1;
    }else{
        max = mid-1;
    }
}
console.log(max);

다른 풀이 코드01.  

function canDistribute(cookieLen, numChildren, targetLen){
    let cnt = 0; 
    for(let length of cookieLen){
        cnt += Math.floor(length/targetLen); 
        if(cnt >= numChildren){
            return true; 
        }
    }
    return false; 
}

function findMaxCookieLength(cookieLen, numChildren){
    let left = 1; 
    let right = Math.max(...cookieLen); 
    let result = 0; 
    
    while(left <= right){
          let mid = Math.floor((left + right)/2); 
            if(canDistribute(cookieLen, numChildren, mid)){
                result = mid; 
                left = mid + 1; 
            }else{
                right = mid -1; 
            }
          }
    return result; 
}
반응형