본문 바로가기
ALGORITHM/프로그래머스 With JS

[프로그래머스] 다리를 지나는 트럭 / Javascript

by LAY CODER 2021. 4. 29.
728x90

문제

 

코딩테스트 연습 - 다리를 지나는 트럭

트럭 여러 대가 강을 가로지르는 일 차선 다리를 정해진 순으로 건너려 합니다. 모든 트럭이 다리를 건너려면 최소 몇 초가 걸리는지 알아내야 합니다. 트럭은 1초에 1만큼 움직이며, 다리 길이

programmers.co.kr

 


 

TRY

function solution(bridge_length, weight, truck_weights) {

    // 다리배열을 다리 길이보다 하나 작게 0으로 채워서 만들고
    // 마지막 요소에 첫 트럭을 넣는다
    let bridge = new Array(bridge_length - 1).fill(0);
    bridge.push(truck_weights.shift());
    // 트럭 하나가 이미 지났다고 가정해서 1초 지난 걸로 초기화
    let time = 1;
    // 다리에 트럭이 없을때 까지 무한루프
    while (bridge.reduce((a, b) => a + b) > 0) {
        // 다리 제일 앞에 있는 요소를 보내고
        // 버틸 수 있는 다리 무게가 다음 트럭보다 높은지 비교한다.
        // 버틸 수 있으면 다리 마지막 요소에 트럭을 넣고 시간을 경과 시키고
        // 버틸 수 없으면 0을 넣고 시간을 경과 시킨다.
        bridge.shift();
        if (weight - bridge.reduce((a, b) => a + b) >= truck_weights[0]) {
            bridge.push(truck_weights.shift());
        } else {
            bridge.push(0);
        }
        time++;
    }
    return time;
}

console.log(solution(2, 10, [7, 4, 5, 6]));
console.log(solution(100, 100, [10]));
console.log(solution(100, 100, [10, 10, 10, 10, 10, 10, 10, 10, 10, 10]));

 

다리 배열을 만들어서 첫 트럭을 넣고 시간을 1로 초기화 한다.

 

Math.max(...array) >  를 이용하여 다리에 트럭이 있는지 확인하며 다리에 첫 번째 요소를 꺼낸다.

 

다리가 감당할 수 있는 무게면 트럭을 넣고 아니면 0을 넣고 시간을 경과 시키는 방식으로 풀이하였다.

 

아무래도 함수가 많이 들어갔기 때문에 속도면에서 좋은 풀이 방법은 아닌거 같다.

 


 

Refactoring

function solution(bridge_length, weight, truck_weights) {
    // qu의 요소 : [트럭무게, 트럭이 빠져 나갈 시간].
    // weightOnBridge : 다리 위 무게
    let time = 0,
        qu = [[0, 0]],
        weightOnBridge = 0;
    // 대기 트럭, 다리를 건너는 트럭이 모두 0일 때 까지 다음 루프 반복
    while (qu.length > 0 || truck_weights.length > 0) {
        // 1. 현재 시간이, 큐 맨 앞의 차의 '나갈 시간'과 같다면 내보내주고,
        //    다리 위 트럭 무게 합에서 빼준다.
        if (qu[0][1] === time) weightOnBridge -= qu.shift()[0];
        if (weightOnBridge + truck_weights[0] <= weight) {
            // 2. 다리 위 트럭 무게 합 + 대기중인 트럭의 첫 무게가 감당 무게 이하면
            //    다리 위 트럭 무게 업데이트, 큐 뒤에 [트럭무게, 이 트럭이 나갈 시간] 추가.
            weightOnBridge += truck_weights[0];
            qu.push([truck_weights.shift(), time + bridge_length]);
        } else {
            // 3. 다음 트럭이 못올라오는 상황이면 얼른 큐의
            //    첫번째 트럭이 빠지도록 그 시간으로 점프한다.
            //    참고: if 밖에서 1 더하기 때문에 -1 해줌
            if (qu[0]) time = qu[0][1] - 1;
        }
        // 시간 업데이트 해준다.
        time++;
    }
    return time;
}

console.log(solution(2, 10, [7, 4, 5, 6]));
console.log(solution(100, 100, [10]));
console.log(solution(100, 100, [10, 10, 10, 10, 10, 10, 10, 10, 10, 10]));

 

위 풀이의 핵심은 트럭이 못 올라오는 상황이면 트럭을 한칸씩 앞으로 전진하면서 기다리는 게 아니고

 

트럭이 빠지도록 하고 그 시간을 더해주기 때문에 트럭이 기다리는 시간을 훨씬 단축 시킬 수 있다.

 

이런식으로 하면 다리가 아무리 길어도 빠르게 시간낭비가 없기 때문에 굉장히 효율적이다.

 

 

 

 

댓글