본문 바로가기

PS/Samsung

[백준/c++] 17779 게리멘더링 2(다시풀어야돼)

문제

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

재현시는 크기가 N×N인 격자로 나타낼 수 있다. 격자의 각 칸은 구역을 의미하고, r행 c열에 있는 구역은 (r, c)로 나타낼 수 있다.  1) 구역을 다섯 개의 선거구로 나눠야 하고, 각 구역은 다섯 선거구 중 하나에 포함되어야 한다.  2) 선거구는 구역을 적어도 하나 포함해야 하고, 한 선거구에 포함되어 있는 구역은 모두 연결되어 있어야 한다. 구역 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]이고, 선거구의 인구는 선거구에 포함된 구역의 인구를 모두 합한 값이다. 선거구를 나누는 방법 중에서, 3) 인구가 가장 많은 선거구와 가장 적은 선거구의 인구 차이의 최솟값을 구해보자.

입력

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

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

출력

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

제한

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

 

구현해야할 조건

기준점 x,y와 경계의 길이 d1,d2를 정한다. d1, d2 ≥ 1, 1 ≤ x < x+d1+d2 ≤ N, 1 ≤ y-d1 < y < y+d2 ≤ N의 범위를 맞춰서 모든 갯수를 구하면 된다.

경계선 부분은 1번은 1번 지역구 경계를 나타내는 부분이고, (x, y), (x+1, y-1), ..., (x+d1, y-d1)
2번은 2번 지역구를 나타내는 부분 (x, y), (x+1, y+1), ..., (x+d2, y+d2)
3번은 3번 지역은 (x+d1, y-d1), (x+d1+1, y-d1+1), ... (x+d1+d2, y-d1+d2)
4번은 4번 지역구로 경계를 맞춰서 구현 (x+d2, y+d2), (x+d2+1, y+d2-1), ..., (x+d2+d1, y+d2-d1) 을 해주면 되는데, 마지막 5번 선거구는 경계에 맞춰서 남은 자리에 5를 넣어주면 된다. 처음에 5를 집어넣고 시작을 해도 무관하다!

경계에 맞춰서 구현을 잘 해야한다.

code

 

#include<iostream>
#include<cstring>
#include<algorithm>

#define MAX 21
#define INF 987654321

using namespace std;

typedef struct{
	int y;
	int x;
}location;

int n;
int res = INF;
int area[MAX][MAX];
int copyArea[MAX][MAX];

location loc[4];

bool ChkLine(int x, int y, int d1, int d2)
{
	/*
	(x, y), (x+1, y-1), ..., (x+d1, y-d1)
	(x, y), (x+1, y+1), ..., (x+d2, y+d2)
	(x+d1, y-d1), (x+d1+1, y-d1+1), ... (x+d1+d2, y-d1+d2)
	(x+d2, y+d2), (x+d2+1, y+d2-1), ..., (x+d2+d1, y+d2-d1)
	*/
	
	if(x + d1 >= n || y - d1 < 0) return false;
	if(x + d2 >= n || y + d2 >= n) return false;
	if(x + d1 + d2 >= n || y - d1 + d2 >= n) return false;
	if(x + d2 + d1 >= n || y + d2 - d1 < 0) return false;
	
	return true;
}

void DFS(int x, int y, int d1, int d2)
{
	for(int i = 0; i < n; i++)
	{
		for(int j = 0; j < n; j++)
		{
			copyArea[i][j] = 5; // 지역구 5 
		}
	}
	
	int decline = 0;
	
	// 1번 : 1 <= r < x + d1, 1 <= c <= y 
	
	for(int i = 0; i < loc[1].x; i++)
	{
		if(i >= loc[0].x)
			decline++;
		for(int j = 0; j <= loc[0].y - decline; j++)
		{
			copyArea[i][j] = 1;
		}
	}
	
	int incline = 0;
	
	// 2번 : 1 <= r <= x + d2 , y < c <= n 
	
	for(int i = 0; i <= loc[2].x; i++)
	{
		if(i > loc[0].x)
			incline++;
		for(int j = loc[0].y + 1 + incline; j < n; j++)
		{
			copyArea[i][j] = 2;
		}
	}
	
	decline = 0;
	
	for(int i = n - 1; i >= loc[1].x; i--)
	{
		if(i < loc[3].x)
			decline++;
		for(int j = 0; j < loc[3].y - decline; j++)
		{
			copyArea[i][j] = 3;
		}
	}
	
	incline = 0;
	
	for(int i = n - 1; i > loc[2].x; i--)
	{
		if(i <= loc[3].x)
			incline++;
		
		for(int j = loc[3].y + incline; j < n; j++)
		{
			copyArea[i][j] = 4;
		}
	}
	
	int people[6] = {0,0,0,0,0,0};
	
	for(int i = 0; i < n; i++)
	{
		for(int j = 0; j < n; j++)
		{
			people[copyArea[i][j]] = people[copyArea[i][j]] + area[i][j];
		}
	}
	
	sort(people, people + 6);
	
	int diff = people[5] - people[1];
	
	res = min(diff, res);
} 


int main()
{
	cin >> n;
	
	for(int i = 0; i < n; i++)
	{
		for(int j = 0; j < n; j++)
		{
			cin >> area[i][j];
		}
	}
	
	for(int i = 0; i < n; i++)
	{
		for(int j = 1; j < n; j++)
		{
			for(int d1 = 1; d1 <= j; d1++)
			{
				for(int d2 = 1; d2 < n - j; d2++)
				{
					if(ChkLine(i,j,d1,d2) == true)
					{
						loc[0].x = i; loc[0].y = j; // 기준 
						loc[1].x = i + d1; loc[1].y = j - d1; // 좌측  지역구 범위 
						loc[2].x = i + d2; loc[2].y = j + d2; // 우측 지역구 
						loc[3].x = i + d1 + d2; loc[3].y = j - d1 + d2; // 하 측 지역구 
						DFS(i,j,d1,d2);
					}
				}
			}
		}
	}
	
	cout << res;
	
	return 0;
}

'PS > Samsung' 카테고리의 다른 글

[백준/c++] 17822 원판 돌리기  (0) 2022.02.15
[백준/c++] 17837 새로운 게임 2  (0) 2022.02.14
[백준/c++] 17142 연구소 3  (0) 2022.02.13
[백준/c++] 17140 이차원 배열과 연산  (0) 2022.02.12
[백준/c++] 17143 낚시왕  (0) 2022.02.11