[Algorithm] BaekJoon : 17779. 게리맨더링 2 by Python

엄희관·2021년 3월 27일
1

Algorithm

목록 보기
118/128
post-thumbnail
post-custom-banner

[문제 바로가기] https://www.acmicpc.net/problem/17779

📌문제 설명

재현시의 시장 구재현은 지난 몇 년간 게리맨더링을 통해서 자신의 당에게 유리하게 선거구를 획정했다. 견제할 권력이 없어진 구재현은 권력을 매우 부당하게 행사했고, 심지어는 시의 이름도 재현시로 변경했다. 이번 선거에서는 최대한 공평하게 선거구를 획정하려고 한다.

재현시는 크기가 N×N인 격자로 나타낼 수 있다. 격자의 각 칸은 구역을 의미하고, r행 c열에 있는 구역은 (r, c)로 나타낼 수 있다. 구역을 다섯 개의 선거구로 나눠야 하고, 각 구역은 다섯 선거구 중 하나에 포함되어야 한다. 선거구는 구역을 적어도 하나 포함해야 하고, 한 선거구에 포함되어 있는 구역은 모두 연결되어 있어야 한다. 구역 A에서 인접한 구역을 통해서 구역 B로 갈 수 있을 때, 두 구역은 연결되어 있다고 한다. 중간에 통하는 인접한 구역은 0개 이상이어야 하고, 모두 같은 선거구에 포함된 구역이어야 한다.

선거구를 나누는 방법은 다음과 같다.

  1. 기준점 (x, y)와 경계의 길이 d1, d2를 정한다. (d1, d2 ≥ 1, 1 ≤ x < x+d1+d2 ≤ N, 1 ≤ y-d1 < y < y+d2 ≤ N)
  2. 다음 칸은 경계선이다.
    1.(x, y), (x+1, y-1), ..., (x+d1, y-d1)
    2.(x, y), (x+1, y+1), ..., (x+d2, y+d2)
    3.(x+d1, y-d1), (x+d1+1, y-d1+1), ... (x+d1+d2, y-d1+d2)
    4.(x+d2, y+d2), (x+d2+1, y+d2-1), ..., (x+d2+d1, y+d2-d1)
  3. 경계선과 경계선의 안에 포함되어있는 곳은 5번 선거구이다.
  4. 5번 선거구에 포함되지 않은 구역 (r, c)의 선거구 번호는 다음 기준을 따른다.
  • 1번 선거구: 1 ≤ r < x+d1, 1 ≤ c ≤ y
  • 2번 선거구: 1 ≤ r ≤ x+d2, y < c ≤ N
  • 3번 선거구: x+d1 ≤ r ≤ N, 1 ≤ c < y-d1+d2
  • 4번 선거구: x+d2 < r ≤ N, y-d1+d2 ≤ c ≤ N

아래는 크기가 7×7인 재현시를 다섯 개의 선거구로 나눈 방법의 예시이다.

구역 (r, c)의 인구는 A[r][c]이고, 선거구의 인구는 선거구에 포함된 구역의 인구를 모두 합한 값이다. 선거구를 나누는 방법 중에서, 인구가 가장 많은 선거구와 가장 적은 선거구의 인구 차이의 최솟값을 구해보자.

입력
첫째 줄에 재현시의 크기 N이 주어진다.

둘째 줄부터 N개의 줄에 N개의 정수가 주어진다. r행 c열의 정수는 A[r][c]를 의미한다.

출력
첫째 줄에 인구가 가장 많은 선거구와 가장 적은 선거구의 인구 차이의 최솟값을 출력한다.

제한

  • 5 ≤ N ≤ 20
  • 1 ≤ A[r][c] ≤ 100

💡 문제 풀이

구현(시뮬레이션) 유형의 문제다. - 범위만 잘 설정하면 쉽게 풀 수 있는 문제다.

문제해결을 위해서는 4개의 값(row, col, d1, d2)을 찾아야 한다.

먼저, 기준점(row, col)의 행과 열은 d1, d2가 최소 1의 값을 가지므로 row는 0 ~ N-2 / col은 1 ~ N-1 까지의 범위가 가능하다는 것을 알 수 있다.

다음으로는 각 기준점에서 구역을 나눌 수 있는 d1, d2를 찾아야 한다.
→ check 함수

d1, d2를 찾기 위해서는 각 구역의 경계선이 되는 지점을 생각하며 조건문으로 처리하였다.
ex) x+d1 / x+d2 / y-d1+d2 등

가능한 기준점(row, col), d1, d2을 정하였으면 5개 구역의 인구수를 계산한다.
구역 1 ~ 4는 경계선에 맞춰 행, 열 값을 조절하며 구하면 되고, 구역 5는 '전체 인구수 - (구역 1 ~ 4)'로 구하였다.

각 구역의 인구수를 구한후 answer에 인구차이의 최소값을 갱신한다.

코드는 다음과 같다.

import sys

N = int(input())
matrix = [list(map(int, sys.stdin.readline().split())) for _ in range(N)]
total = 0
for row in matrix:
    total += sum(row)
answer = int(1e9)

def calculate(row, col, d1, d2):
    global total, answer
    first, second, third, fourth = 0, 0, 0, 0

    # 구역 1
    col1 = col+1
    for r in range(row+d1):
        if r >= row:
            col1 -= 1
        first += sum(matrix[r][:col1])

    # 구역 2
    col2 = col+1
    for r in range(row+d2+1):
        if r > row:
            col2 += 1
        second += sum(matrix[r][col2:])
   
    # 구역 3
    col3 = col - d1
    for r in range(row+d1, N):
        third += sum(matrix[r][:col3])
        if r < row+d1+d2:
            col3 += 1

    # 구역 4
    col4 = (col+d2) - N
    for r in range(row+d2+1, N):
        fourth += sum(matrix[r][col4:])
        if r <= row+d1+d2:
            col4 -= 1
            
    # 구역 5
    five = total - first - second - third - fourth
    answer = min(answer, (max(first, second, third, fourth, five) - min(first, second, third, fourth, five)))

def check(r, c, d1, d2): # 가능한 d1, d2 찾기
    if 0 <= r+d1-1 < N and 0 <= r+d2-1 < N and 0 <= c-d1+d2-1 < N:
        if 0 <= c-d1 and c+d2 < N and r+d1+d2 < N:
            calculate(r, c, d1, d2)

for r in range(N-2):
    for c in range(1, N-1):
        for d1 in range(1, N-1):
            for d2 in range(1, N-1):
                check(r, c, d1, d2)
print(answer)

profile
허브
post-custom-banner

1개의 댓글

comment-user-thumbnail
2022년 10월 9일

안녕하세요.. 포스트에서
먼저, 기준점(row, col)의 행과 열은 d1, d2가 최소 1의 값을 가지므로 row는 0 ~ N-2 / col은 1 ~ N-1 까지의 범위가 가능하다는 것을 알 수 있다.
혹시 이거 무슨 뜻인지 알수있을까요..?
choi_kh01@naver.com 으로 알려주시면 감사하겠습니다..!

답글 달기