프로그래머스 - 전력망을 둘로 나누기 (위클리 챌린지 9주차)

아놀드·2021년 10월 5일
1

프로그래머스

목록 보기
41/52
post-custom-banner

1. 문제

문제 링크

설명

n개의 송전탑이 전선을 통해 하나의 트리 형태로 연결되어 있습니다. 당신은 이 전선들 중 하나를 끊어서 현재의 전력망 네트워크를 2개로 분할하려고 합니다. 이때, 두 전력망이 갖게 되는 송전탑의 개수를 최대한 비슷하게 맞추고자 합니다.

송전탑의 개수 n, 그리고 전선 정보 wires가 매개변수로 주어집니다. 전선들 중 하나를 끊어서 송전탑 개수가 가능한 비슷하도록 두 전력망으로 나누었을 때, 두 전력망이 가지고 있는 송전탑 개수의 차이(절대값)를 return 하도록 solution 함수를 완성해주세요.

제한 사항

  • n은 2 이상 100 이하인 자연수입니다.
  • wires는 길이가 n-1인 정수형 2차원 배열입니다.
    • wires의 각 원소는 [v1, v2] 2개의 자연수로 이루어져 있으며, 이는 전력망의 v1번 송전탑과 v2번 송전탑이 전선으로 연결되어 있다는 것을 의미합니다.
    • 1 ≤ v1 < v2 ≤ n 입니다.
    • 전력망 네트워크가 하나의 트리 형태가 아닌 경우는 입력으로 주어지지 않습니다.

입출력 예

nwiresresult
9[[1,3],[2,3],[3,4],[4,5],[4,6],[4,7],[7,8],[7,9]]3
4[[1,2],[2,3],[3,4]]0
7[[1,2],[2,7],[3,7],[3,4],[4,5],[6,7]]1

입출력 예 설명

입출력 예 #1

  • 4번과 7번을 연결하는 전선을 끊으면 두 전력망은 각 6개와 3개의 송전탑을 가지며, 이보다 더 비슷한 개수로 전력망을 나눌 수 없습니다.
  • 또 다른 방법으로는 3번과 4번을 연결하는 전선을 끊어도 최선의 정답을 도출할 수 있습니다.

2. 풀이

잡담
위클리 문제도 벌써 9주차가 됐습니다.
조회수좀 끌어보겠다고 제일 핫한 3주차 문제를 포스팅했을 때가 엊그제 같은데
벌써 7주 전이네요... 이럴 때마다 시간을 정말 응집력있게 써야겠다고 생각이 듭니다.
시간을 그냥저냥 보내면 한 달, 석 달 뒤에 그 시간을 돌아봤을 때,
빠르게 지나갔다고 느껴지고 '그동안 뭐했나' 생각이 들며 약간의 자괴감이 들기도 하지만
시간을 값지게 사용하면 석 달 뒤에 그 시간을 돌아봤을 때,
'내가 석 달 전에 비해 이만큼 성장했구나', '조금 더 열심히 해야겠다' 생각하며
점점 더 능동적인 사람이 되어가는 듯 합니다.

본 풀이
각설하고 이번에도 위클리 문제를 풀어보겠습니다.
왜냐하면 위클리 문제가 조회수를 많이 뽑을 수 있기 때문입니다.

이 문제는 그래프 탐색의 기본기를 물어보는 문제인데요,
푸는 방식은 총 4가지로 구분될 수 있습니다.

  1. 인접 리스트와 BFS 사용
  2. 인접 리스트와 DFS 사용
  3. 인접 배열과 BFS 사용
  4. 인접 배열과 DFS 사용

저는 4번 방법으로 풀어보겠습니다.

계획1 - 인접 배열 초기화

// 2차원 배열 생성
const adjArr = [...Array(n + 1)].map(() => [...Array(n + 1)].map(() => 0));
// 인접 배열 초기화
wires.forEach(([v1, v2]) => {
   adjArr[v1][v2] = 1;
   adjArr[v2][v1] = 1;
});

그래프 탐색의 기본입니다.

계획2 - 전선을 하나씩 다 끊어보며 순회합니다.

// 순회
const dfs = (tower) => {
    visit[tower] = 1;
    count++;

    for (let i = 1; i <= n; i++) {
        adjArr[tower][i] && !visit[i] && dfs(i);
    }
};

return wires.reduce((m, [v1, v2]) => {
    // 전선 끊기
    adjArr[v1][v2] = 0;
    adjArr[v2][v1] = 0;
    // 순회
    dfs(1);
    // 전선 회복
    adjArr[v1][v2] = 1;
    adjArr[v2][v1] = 1;

    m = Math.min(m, Math.abs(n - 2 * count)); // 송전탑 개수의 차이의 최솟값 갱신
    visit.forEach((_, i) => visit[i] = 0); // 방문 배열 초기화
    count = 0; // 개수 초기화

    return m;
}, n);

하나의 연결된 전력망만 순회하며 통신탑의 개수만 새면
다른 하나의 전력망의 통신탑의 개수를 알 수 있습니다.

현재 전력망을 순회하면서 카운팅한 통신탑의 개수를 count라 했을 때,
다른 전력망의 통신탑의 개수는 n - count가 됩니다.
이 둘의 통신탑 개수의 차이는 n - count - count가 되고
절댓값을 취해야하니 Math.abs(n - 2 * count) 코드가 들어가게 됩니다.

3. 전체 코드

function solution(n, wires) {
    const adjArr = [...Array(n + 1)].map(() => [...Array(n + 1)].map(() => 0));
    const visit = Array(n + 1).fill(0);
    let count = 0;

    wires.forEach(([v1, v2]) => {
        adjArr[v1][v2] = 1;
        adjArr[v2][v1] = 1;
    });

    // 순회
    const dfs = (tower) => {
        visit[tower] = 1;
        count++;

        for (let i = 1; i <= n; i++) {
            adjArr[tower][i] && !visit[i] && dfs(i);
        }
    };

    return wires.reduce((m, [v1, v2]) => {
        // 전선 끊기
        adjArr[v1][v2] = 0;
        adjArr[v2][v1] = 0;
        // 순회
        dfs(1);
        // 전선 회복
        adjArr[v1][v2] = 1;
        adjArr[v2][v1] = 1;

        m = Math.min(m, Math.abs(n - 2 * count)); // 송전탑 개수의 차이의 최솟값 갱신
        visit.forEach((_, i) => visit[i] = 0); // 방문 배열 초기화
        count = 0; // 개수 초기화

        return m;
    }, n);
}
profile
함수형 프로그래밍, 자바스크립트에 관심이 많습니다.
post-custom-banner

2개의 댓글

comment-user-thumbnail
2021년 10월 6일

안녕하세요, 지나가다가 코드 풀이 정말 인상깊게 봤습니다. 내용을 읽다보니 개인적으로 생각난 의견들이 있어서 댓글로 남겨봅니다.

  1. 송전탑이 트리 구조인데 visit 배열이 왜 필요한가요? 트리의 순회는 사이클이 일어나지 않는데, 방문 노드를 기록할 필요가 있는지 궁금합니다.
  2. count를 구하는 것을 1번 송전탑으로 고정하지 않고 wire의 양 옆 송전탑의 count를 각각 구하면, 트리의 순회가 단방향임을 이용해서 메모라이징 기법을 사용할 수 있지 않을까요?
1개의 답글