알고리즘

BOJ 23291 - 어항 정리(Java)

장중앙 2021. 12. 22. 14:03

https://www.acmicpc.net/problem/23291

 

23291번: 어항 정리

마법사 상어는 그동안 배운 마법을 이용해 어항을 정리하려고 한다. 어항은 정육면체 모양이고, 한 변의 길이는 모두 1이다. 상어가 가지고 있는 어항은 N개이고, 가장 처음에 어항은 일렬로 바

www.acmicpc.net

 

풀이

물고기 수의 최대/최소인 어항의 차이가 K이하일 때 까지 반복

 

1. 물고기가 최소인 어항에 물고기 추가

    - 어항에 물고기 수가 최소인 어항들에 물고기 수를 1씩 추가

2. 어항 쌓기

   - 시계 방향 90도로 어항을 말아서 쌓음 

   - 처음에는 가장 왼쪽의 1개 항만 이동, 이후부터는 쌓여져있는 모든 어항을 이동

3. 물고기 수 조절

   - 상하좌우 인접 어항을 확인, 현재 어항이 인접한 어항의 물고기 수와 비교

   - (현재 어항 - 이동할 어항)/5 >0 이라면 물고기 이동(물고기 이동은 모든 어항에서 동시에 발생)

4. 어항 일렬로 배치

    - 쌓여진 어항을 일렬로 재배치

    - 가장 왼쪽에 있는 어항부터 밑에서 위의 순서로 배치

5. 어항 접기

    - 2번 반복

    - 가운데를 중심으로 왼쪽을 180도 회전시켜 오른쪽 위에 배치

6. 물고기 수 조절

7. 어항 일렬로 배치 

 

 

어항 쌓기

private static void make2D() {
	int pivotX = 1;
	int w = 1;
	int h = 1;
	int idx = 0;
	while (pivotX - 1 + w + h <= N) {
		idx++;
		for (int x = pivotX; x < pivotX + w; x++) {
			for (int y = N; y > N - h; y--) {
				int ny = N - w + x - pivotX;
				int nx = pivotX + w + N - y;
				map[ny][nx] = map[y][x];
				map[y][x] = 0;
			}
		}
		pivotX += w;
		if (idx % 2 == 0) {
			w++;
		} else {
			h++;
		}
	}
}

 

물고기 조절

물고기의 이동은 동시에 진행되기 때문에 물고기의 이동을 저장하기 위한 배열 생성

물고기가 존재하는 위치마다 상하좌우의 어항을 확인하여 물고기가 있다면 물고기 이동 조건에 해당하는 지 확인, 이동 수를 배열에 누적 갱신

물고기 이동 배열을 완전히 갱신한 후에 배열의 각 요소를 이동수만큼 ± 계산

private static void adjustFish() {
	int[][] adjustMap = new int[N + 1][N + 1];

	for (int y = 1; y <= N; y++) {
		for (int x = 1; x <= N; x++) {
			if (map[y][x] == 0)
				continue;
			for (int d = 0; d < 4; d++) {
				int ny = y + dy[d];
				int nx = x + dx[d];
				if ((ny < 1 || nx < 1 || ny > N || nx > N) || map[ny][nx] == 0)
					continue;
				int diff = map[y][x] - map[ny][nx];
				diff /= 5;
				if (diff > 0) {
					adjustMap[y][x] -= diff;
					adjustMap[ny][nx] += diff;
				}
			}
		}
	}
	for (int y = 1; y <= N; y++) {
		for (int x = 1; x <= N; x++) {
			map[y][x] += adjustMap[y][x];
		}
	}
}

 

 

어항 일렬 배치

private static void make1D() {
	List<Integer> list = new ArrayList<>();
	for (int x = 1; x <= N; x++) {
		for (int y = N; y >= 1; y--) {
			if (map[y][x] == 0)
				break;
			list.add(map[y][x]);
			map[y][x] = 0;
		}
	}
	for (int i = 0; i < N; i++) {
		map[N][i + 1] = list.get(i);
	}
}

 

어항 접기

접기는 2번 반복 N -> N/2 -> N/4

private static void fold() {
	List<Integer> list =new ArrayList<>();
	int pivotX = 1;
	int yCnt = 1;
	for (int cnt = 1; cnt <= 2; cnt++) {
		int _y= N - yCnt * 2 + 1;
		for (int y = N; y > N-yCnt; y--) {
			list.clear();
			for (int x = pivotX; x < pivotX +(N - pivotX + 1) / 2; x++) {
				list.add(map[y][x]);
				map[y][x]=0;
			}
			for(int idx=0;idx<list.size();idx++) {
				map[_y][N-idx]=list.get(idx);
			}
			_y++;
		}
		yCnt *= 2;
		pivotX += N / 2;
	}
}

 

 

 

전체 코드

import java.io.*;
import java.util.*;

public class Main {
	static final int INF = 987654321;
	static int N, K;
	static int[][] map;
	static int[] dy = { 0, 1, 0, -1 };
	static int[] dx = { 1, 0, -1, 0 };

	public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));

		StringTokenizer st = new StringTokenizer(br.readLine());
		N = Integer.parseInt(st.nextToken());
		K = Integer.parseInt(st.nextToken());

		map = new int[N + 1][N + 1];
		st = new StringTokenizer(br.readLine());
		for (int i = 1; i <= N; i++) {
			map[N][i] = Integer.parseInt(st.nextToken());
		}

		int time = 0;
		while (!isFinish()) {
			time++;
			pushFish(); 
			
			make2D();
			adjustFish();
			
			make1D();
			fold();
			adjustFish();
			
			make1D();
		}
		System.out.println(time);
	}

	private static boolean isFinish() {
		int max = 0;
		int min = INF;
		for (int y = 1; y <= N; y++) {
			for (int x = 1; x <= N; x++) {
				if (map[y][x] == 0)
					continue;
				if (max < map[y][x])
					max = map[y][x];
				if (min > map[y][x])
					min = map[y][x];
			}
		}
		if (max - min <= K) {
			return true;
		} else
			return false;
	}

	private static void fold() {
		// 2번 반복
		List<Integer> list =new ArrayList<>();
		int pivotX = 1;
		int yCnt = 1;
		for (int cnt = 1; cnt <= 2; cnt++) {
			int _y=N-yCnt*2+1;
			for (int y = N; y > N-yCnt; y--) {
				
				list.clear();
				for (int x = pivotX; x < pivotX +(N - pivotX + 1) / 2; x++) {
					list.add(map[y][x]);
					map[y][x]=0;
				}
				for(int idx=0;idx<list.size();idx++) {
					map[_y][N-idx]=list.get(idx);
				}
				_y++;
			}
			yCnt *= 2;
			pivotX += N / 2;
		}

	}

	private static void make1D() {
		List<Integer> list = new ArrayList<>();
		for (int x = 1; x <= N; x++) {
			for (int y = N; y >= 1; y--) {
				if (map[y][x] == 0)
					break;
				list.add(map[y][x]);
				map[y][x] = 0;
			}
		}
		for (int i = 0; i < N; i++) {
			map[N][i + 1] = list.get(i);
		}
	}

	private static void make2D() {
		int pivotX = 1;
		int w = 1;
		int h = 1;
		int idx = 0;
		while (pivotX - 1 + w + h <= N) {
			idx++;
			for (int x = pivotX; x < pivotX + w; x++) {
				for (int y = N; y > N - h; y--) {
					int ny = N - w + x - pivotX;
					int nx = pivotX + w + N - y;
					map[ny][nx] = map[y][x];
					map[y][x] = 0;
				}
			}
			pivotX += w;
			if (idx % 2 == 0) {
				w++;
			} else {
				h++;
			}
		}
	}

	// 물고기 수가 가장 적은 어항에 물고기 추가
	private static void pushFish() {
		List<Integer> posList = new ArrayList<>();
		int min = INF;
		for (int i = 1; i <= N; i++) {
			if (map[N][i] < min) {
				min = map[N][i];
				posList.clear();
				posList.add(i);
			} else if (map[N][i] == min) {
				posList.add(i);
			}
		}

		for (int idx : posList) {
			map[N][idx]++;
		}
	}

	private static void adjustFish() {
		int[][] adjustMap = new int[N + 1][N + 1];

		for (int y = 1; y <= N; y++) {
			for (int x = 1; x <= N; x++) {
				if (map[y][x] == 0)
					continue;
				for (int d = 0; d < 4; d++) {
					int ny = y + dy[d];
					int nx = x + dx[d];
					if ((ny < 1 || nx < 1 || ny > N || nx > N) || map[ny][nx] == 0)
						continue;
					int diff = map[y][x] - map[ny][nx];
					diff /= 5;
					// 현재 위치에서 다음 위치로 물고기 보내기 가능
					if (diff > 0) {
						adjustMap[y][x] -= diff;
						adjustMap[ny][nx] += diff;
					}
				}
			}
		}
		// 보내고 받는 물고기 수를 +- 계산
		for (int y = 1; y <= N; y++) {
			for (int x = 1; x <= N; x++) {
				map[y][x] += adjustMap[y][x];
			}
		}
	}

}