[프로그래머스] 택배 배달과 수거하기 (js)

2024. 11. 17. 23:20· 코테 준비/프로그래머스
목차
  1. 문제 설명
  2. 나의 풀이

문제 설명

당신은 일렬로 나열된 n개의 집에 택배를 배달하려 합니다. 배달할 물건은 모두 크기가 같은 재활용 택배 상자에 담아 배달하며, 배달을 다니면서 빈 재활용 택배 상자들을 수거하려 합니다.
배달할 택배들은 모두 재활용 택배 상자에 담겨서 물류창고에 보관되어 있고, i번째 집은 물류창고에서 거리 i만큼 떨어져 있습니다. 또한 i번째 집은 j번째 집과 거리 j - i만큼 떨어져 있습니다. (1 ≤ i ≤ j ≤ n)
트럭에는 재활용 택배 상자를 최대 cap개 실을 수 있습니다. 트럭은 배달할 재활용 택배 상자들을 실어 물류창고에서 출발해 각 집에 배달하면서, 빈 재활용 택배 상자들을 수거해 물류창고에 내립니다. 각 집마다 배달할 재활용 택배 상자의 개수와 수거할 빈 재활용 택배 상자의 개수를 알고 있을 때, 트럭 하나로 모든 배달과 수거를 마치고 물류창고까지 돌아올 수 있는 최소 이동 거리를 구하려 합니다. 각 집에 배달 및 수거할 때, 원하는 개수만큼 택배를 배달 및 수거할 수 있습니다.

다음은 cap=4 일 때, 최소 거리로 이동하면서 5개의 집에 배달 및 수거하는 과정을 나타낸 예시입니다.

배달 및 수거할 재활용 택배 상자 개수


집 #1 집 #2 집 #3 집 #4 집 #5
배달 1개 0개 3개 1개 2개
수거 0개 3개 0개 4개 0개

배달 및 수거 과정


집 #1 집 #2 집 #3 집 #4 집 #5 설명
남은 배달/수거 1/0 0/3 3/0 1/4 2/0 물류창고에서 택배 3개를 트럭에 실어 출발합니다.
남은 배달/수거 1/0 0/3 3/0 0/4 0/0 물류창고에서 5번째 집까지 이동하면서(거리 5) 4번째 집에 택배 1개를 배달하고, 5번째 집에 택배 2개를 배달합니다.
남은 배달/수거 1/0 0/3 3/0 0/0 0/0 5번째 집에서 물류창고까지 이동하면서(거리 5) 4번째 집에서 빈 택배 상자 4개를 수거한 후, 수거한 빈 택배 상자를 물류창고에 내리고 택배 4개를 트럭에 싣습니다.
남은 배달/수거 0/0 0/3 0/0 0/0 0/0 물류창고에서 3번째 집까지 이동하면서(거리 3) 1번째 집에 택배 1개를 배달하고, 3번째 집에 택배 3개를 배달합니다.
남은 배달/수거 0/0 0/0 0/0 0/0 0/0 3번째 집에서 물류창고까지 이동하면서(거리 3) 2번째 집에서 빈 택배 상자 3개를 수거한 후, 수거한 빈 택배 상자를 물류창고에 내립니다.

16(=5+5+3+3)의 거리를 이동하면서 모든 배달 및 수거를 마쳤습니다. 같은 거리로 모든 배달 및 수거를 마치는 다른 방법이 있지만, 이보다 짧은 거리로 모든 배달 및 수거를 마치는 방법은 없습니다.

트럭에 실을 수 있는 재활용 택배 상자의 최대 개수를 나타내는 정수 cap, 배달할 집의 개수를 나타내는 정수 n, 각 집에 배달할 재활용 택배 상자의 개수를 담은 1차원 정수 배열 deliveries와 각 집에서 수거할 빈 재활용 택배 상자의 개수를 담은 1차원 정수 배열 pickups가 매개변수로 주어집니다. 이때, 트럭 하나로 모든 배달과 수거를 마치고 물류창고까지 돌아올 수 있는 최소 이동 거리를 return 하도록 solution 함수를 완성해 주세요.

 

제한사항

  • 1 ≤ cap ≤ 50
  • 1 ≤ n ≤ 100,000
  • deliveries의 길이 = pickups의 길이 = n
  • 트럭의 초기 위치는 물류창고입니다.

 

입출력 예

cap n deliveries pickups result
4 5 [1, 0, 3, 1, 2] [0, 3, 0, 4, 0] 16
2 7 [1, 0, 2, 0, 1, 0, 2] [0, 2, 0, 1, 0, 2, 0] 30

입출력 예 설명

입출력 예 #1

  • 문제 예시와 동일합니다.

입출력 예 #2

배달 및 수거할 재활용 택배 상자 개수


집 #1 집 #2 집 #3 집 #4 집 #5 집 #6 집 #7
배달 1개 0개 2개 0개 1개 0개 2개
수거 0개 2개 0개 1개 0개 2개 0개

배달 및 수거 과정


집 #1 집 #2 집 #3 집 #4 집 #5 집 #6 집 #7 설명
남은 배달/수거 1/0 0/2 2/0 0/1 1/0 0/2 2/0 물류창고에서 택배 2개를 트럭에 실어 출발합니다.
남은 배달/수거 1/0 0/2 2/0 0/1 1/0 0/2 0/0 물류창고에서 7번째 집까지 이동하면서(거리 7) 7번째 집에 택배 2개를 배달합니다.
남은 배달/수거 1/0 0/2 2/0 0/1 1/0 0/0 0/0 7번째 집에서 물류창고까지 이동하면서(거리 7) 6번째 집에서 빈 택배 상자 2개를 수거한 후, 수거한 빈 택배 상자를 물류창고에 내리고 택배 2개를 트럭에 싣습니다.
남은 배달/수거 1/0 0/2 1/0 0/1 0/0 0/0 0/0 물류창고에서 5번째 집까지 이동하면서(거리 5) 3번째 집에 택배 1개를 배달하고, 5번째 집에 택배 1개를 배달합니다.
남은 배달/수거 1/0 0/1 1/0 0/0 0/0 0/0 0/0 5번째 집에서 물류창고까지 이동하면서(거리 5) 4번째 집에서 빈 택배 상자 1개를 수거하고 2번째 집에서 빈 택배 상자 1개를 수거한 후, 수거한 빈 택배 상자를 물류창고에 내리고 택배 2개를 트럭에 싣습니다.
남은 배달/수거 0/0 0/1 0/0 0/0 0/0 0/0 0/0 물류창고에서 3번째 집까지 이동하면서(거리 3) 1번째 집에 택배 1개를 배달하고, 3번째 집에 택배 1개를 배달합니다.
남은 배달/수거 0/0 0/0 0/0 0/0 0/0 0/0 0/0 3번째 집에서 물류창고까지 이동하면서(거리 3) 2번째 집에서 빈 택배 상자 1개를 수거한 후, 수거한 빈 택배 상자를 물류창고에 내립니다.

30(=7+7+5+5+3+3)의 거리를 이동하면서 모든 배달 및 수거를 마쳤습니다. 같은 거리로 모든 배달 및 수거를 마치는 다른 방법이 있지만, 이보다 짧은 거리로 모든 배달 및 수거를 마치는 방법은 없습니다.
따라서, 30을 return 하면 됩니다.

 

나의 풀이

function solution(cap, n, deliveries, pickups) {
    let totalDistance = 0;

    // 현재 트럭이 처리할 마지막 지점
    let deliveryIdx = n - 1;
    let pickupIdx = n - 1;

    while (deliveryIdx >= 0 || pickupIdx >= 0) {
        let currentCap = cap;
        let farthest = 0;

        // 가장 먼 배달 지점 처리
        while (deliveryIdx >= 0 && deliveries[deliveryIdx] === 0) {
            deliveryIdx--;
        }
        farthest = Math.max(farthest, deliveryIdx + 1);

        // 배달 처리
        while (deliveryIdx >= 0 && currentCap > 0) {
            if (deliveries[deliveryIdx] <= currentCap) {
                currentCap -= deliveries[deliveryIdx];
                deliveries[deliveryIdx] = 0;
                deliveryIdx--;
            } else {
                deliveries[deliveryIdx] -= currentCap;
                currentCap = 0;
            }
        }

        // 현재 트럭 용량 초기화
        currentCap = cap;

        // 가장 먼 수거 지점 처리
        while (pickupIdx >= 0 && pickups[pickupIdx] === 0) {
            pickupIdx--;
        }
        farthest = Math.max(farthest, pickupIdx + 1);

        // 수거 처리
        while (pickupIdx >= 0 && currentCap > 0) {
            if (pickups[pickupIdx] <= currentCap) {
                currentCap -= pickups[pickupIdx];
                pickups[pickupIdx] = 0;
                pickupIdx--;
            } else {
                pickups[pickupIdx] -= currentCap;
                currentCap = 0;
            }
        }

        // 가장 먼 지점까지 왕복 거리 추가
        totalDistance += farthest * 2;
    }

    return totalDistance;
}

 

문제를 읽고 그리디 알고리즘을 적용해야겠다고 생각했다. 배달과 수거 작업을 효율적으로 처리하려면 가장 먼 지점부터 작업을 처리하며 이동 거리를 줄여나가는 전략이 적합하다고 판단했기 때문이다.

1. 가장 먼 지점을 우선 처리
배달과 수거 작업을 효율적으로 하기 위해, 배달(deliveries)과 수거(pickups) 배열에서 남은 물건이 있는 가장 먼 지점을 먼저 방문해  트럭이 한 번 이동할 때 최대한 많은 작업을 처리하도록 함 

2. 트럭 용량에 따른 작업
트럭의 용량(cap)을 초과하지 않도록 작업을 진행하며, 한 번에 처리 가능한 만큼만 물건을 배달하거나 수거
작업을 완료한 지점은 배열 값을 갱신하여 해당 지점을 다시 방문하지 않도록 처리함 

3. 왕복 거리 계산
배달 및 수거 작업 중 가장 먼 지점(farthest)을 기준으로 트럭이 이동해야 하는 왕복 거리(farthest * 2)를 계산하고, 이를 반복적으로 누적하여 총 이동 거리를 구함 


'코테 준비 > 프로그래머스' 카테고리의 다른 글

[프로그래머스] 리코쳇 로봇 ( js)  (0) 2024.11.24
[프로그래머스] 숫자 변환하기 (js)  (1) 2024.11.23
[프로그래머스] [PCCP 기출문제] 3번 / 충돌위험 찾기 (js)  (0) 2024.11.16
[프로그래머스] [3차] 자동완성 (js)  (3) 2024.11.13
[프로그래머스][PCCP 기출문제] 2번 / 퍼즐 게임 챌린지 (js)  (1) 2024.11.10
  1. 문제 설명
  2. 나의 풀이
'코테 준비/프로그래머스' 카테고리의 다른 글
  • [프로그래머스] 리코쳇 로봇 ( js)
  • [프로그래머스] 숫자 변환하기 (js)
  • [프로그래머스] [PCCP 기출문제] 3번 / 충돌위험 찾기 (js)
  • [프로그래머스] [3차] 자동완성 (js)
배트리버
배트리버
🐾 사람 좋아, 개발 좋아 🐾 궁금한 건 끝까지 파고들고, 배운 건 즐겁게 나누는 개발자의 놀이터
배트리버
리트리버의 개발 놀이터
배트리버
전체
오늘
어제
  • 분류 전체보기
    • 네트워크
    • 기초 셋팅
    • 오늘의 일기
    • 리액트
    • 코테 준비
      • 프로그래머스
      • 백준
    • 코드스테이츠44기 프론트엔드
    • HTML-CSS-JavaScript
      • HTML
      • CSS
      • JavaScript
    • 자료구조&알고리즘
    • TypeScript
    • Git
    • Tip
    • 프로젝트
    • Next.js
    • 트러블슈팅

블로그 메뉴

  • 홈
  • 태그
  • 방명록

공지사항

인기 글

태그

  • 타입스크립트문법
  • KPT 회고
  • BFS
  • 코드스테이츠 44기
  • 코드스테이츠 44기 프론트엔드
  • 코드스테이츠
  • 리액트 상태관리
  • 코드스테이츠 블로깅
  • 리액트쿼리
  • 프로젝트 회고
  • 리액트
  • 티스토리챌린지
  • 자바스크립트 비동기
  • 코드스테이츠 회고록
  • 탄스택쿼리
  • 프로그래머스
  • 코드스테이츠 프론트엔드
  • 네트워크
  • 자바스크립트
  • 오블완

최근 댓글

최근 글

hELLO · Designed By 정상우.v4.2.2
배트리버
[프로그래머스] 택배 배달과 수거하기 (js)
상단으로

티스토리툴바

단축키

내 블로그

내 블로그 - 관리자 홈 전환
Q
Q
새 글 쓰기
W
W

블로그 게시글

글 수정 (권한 있는 경우)
E
E
댓글 영역으로 이동
C
C

모든 영역

이 페이지의 URL 복사
S
S
맨 위로 이동
T
T
티스토리 홈 이동
H
H
단축키 안내
Shift + /
⇧ + /

* 단축키는 한글/영문 대소문자로 이용 가능하며, 티스토리 기본 도메인에서만 동작합니다.