알고리즘/백준

[백준 17471] 게리맨더링 C++

겜도리도리 2021. 9. 29. 21:14
반응형

문제

백준 17471 게리맨더링 C++

 

17471번: 게리맨더링

선거구를 [1, 4], [2, 3, 5, 6]으로 나누면 각 선거구의 인구는 9, 8이 된다. 인구 차이는 1이고, 이 값보다 더 작은 값으로 선거구를 나눌 수는 없다.

www.acmicpc.net

풀이

N까지의 노드를 0과 1로 양분합니다.

0과 1을 양분할 수 있는 모든 경우의 수에 대해 실행합니다. (조합 이용)

0과 1로 양분할 수 있게 영역이 결정되면, 각 영역을 서로 0은 0끼리, 1은 1끼리 연결할 수 있는지 확인힙니다.

연결이 가능하다면, 각각의 합을 구한 후 차를 구하고, 최솟값과 비교하여 갱신합니다.

소스 코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#include <iostream>
#include <vector>
#include <cstring>
#include <algorithm>
#include <queue>
using namespace std;
 
int popul[11];
int N;
int zerosum, onesum;
bool graph[11][11];
bool divide[11];
bool visited[11];
int minsub = 987654321;
 
// 연결되어 있는지 확인
bool checkConnect(int curnode, bool division, int target) {
    memset(visited, 0sizeof(visited));
    queue<int> q;
    q.push(curnode);
    visited[curnode] = true;
    int t_size = 0;
    while (!q.empty()) {
        int cur = q.front();
        t_size++;
        if (division == false) {
            zerosum += popul[cur];
        }
        else {
            onesum += popul[cur];
        }
        q.pop();
        for (int i = 1; i <= N; i++) {
            if (graph[cur][i] == true && !visited[i] && divide[i] == division)
            {
                visited[i] = true;
                q.push(i);
            }
        }
    }
    if (t_size == target)
        return true;
    else
        return false;
}
 
void selectnode(int node, int cnt, int size) {
    // size만큼 고르면 0, 1끼리 연결되는지 검사
    if (cnt == size) {
        int zerostart = 0, onestart = 0;
        bool isConnectZero, isConnectOne;
        // 0 영역의 스타트노드
        for (int i = 1; i <= N; i++) {
            if (divide[i] == false) {
                zerostart = i;
                break;
            }
        }
        // 1 영역의 스타트노드
        for (int i = 1; i <= N; i++) {
            if (divide[i] == true) {
                onestart = i;
                break;
            }
        }
        // 0 검사
        zerosum = 0;
        isConnectZero = checkConnect(zerostart, 0, N - size);
        // 1 검사
        onesum = 0;
        isConnectOne = checkConnect(onestart, 1size);
        // 둘 다 연결되어 있으면 최솟값 갱신
        if (isConnectOne && isConnectZero) {
            minsub = min(minsub, abs(zerosum - onesum));
        }
        return;
    }
    // size만큼 고를 때 까지 반복
    for (int i = node; i <= N; i++) {
        if (divide[i] == true)
            continue;
        divide[i] = true;
        selectnode(i + 1, cnt + 1size);
        divide[i] = false;
    }
}
void fun() {
    for (int i = 1; i <= N; i++) {
        // 노드들을 0, 1로 나눔
        selectnode(10, i);
    }
    if (minsub == 987654321)
        cout << -1;
    else
        cout << minsub;
}
 
int main() {
    // 입력
    cin >> N;
    for (int i = 1; i <= N; i++) {
        cin >> popul[i];
    }
    // 그래프 연결
    for (int i = 1; i <= N; i++) {
        int num;
        cin >> num;
        for (int j = 0; j < num; j++) {
            int node;
            cin >> node;
            graph[i][node] = true;
        }
    }
    fun();
    return 0;
}
cs
반응형

'알고리즘 > 백준' 카테고리의 다른 글

[백준 1038] 감소하는 수 C++  (0) 2021.10.01
[백준 6593] 상범 빌딩 C++  (0) 2021.10.01
[백준 2668] 숫자고르기 C++  (0) 2021.09.28
[백준 9084] 동전 (C++)  (0) 2021.09.27
[백준 2981] 검문 C++  (0) 2021.09.26