-
백준 - 16236번 아기상어코테 풀이 2024. 1. 19. 12:19
이 문제는 여러 조건들을 만족시키는 메소드를 기능 별로 분리해 만들어야 헷갈리지 않는다.
이 문제의 핵심은 "거리가 가까운 물고기가 많다면, 가장 위에 있는 물고기, 그러한 물고기가 여러마리라면, 가장 왼쪽에 있는 물고기를 먹는다." 를 어떻게 처리하느냐다.
나는 기능 별로 두가지 메소드로 나누었다.
1. bfs 메소드 : 다음에 먹을 가장 적절한 물고기의 위치와 거기로 가는데 걸린 시간 반환
2. move 메소드 : bfs 메소드를 호출해서 물고기를 먹고 time을 업데이트하는 재귀함수
나는 핵심 조건을 bfs 메소드에서 PriorityQueue를 이용해 처리했다.
PriorityQueue<int[]> queue = new PriorityQueue<int[]>(new Comparator<int[]>() { @Override public int compare(int[] a, int[] b) { //시간이 같을 때 if(Integer.compare(a[0],b[0])==0) { //x좌표 같을 때 if(Integer.compare(a[1],b[1])==0) { //y좌표 같을 때 return Integer.compare(a[2],b[2]); } return Integer.compare(a[1],b[1]); } return Integer.compare(a[0],b[0]); } });
시간이 같을 때는 x좌표(map에서 위아래)를 비교하고 x좌표가 같을 때는 y좌표를 비교해서 정렬하도록 했다.
또 하나 유의할 점이, 아기 상어가 자기 자신을 먹을 수는 없기 때문에 크기가 9인 곳은 안 먹는다는 설정을 해야한다.
if(map[cur_i][cur_j]>0 && map[cur_i][cur_j] < size && map[cur_i][cur_j] != 9) { //먹을 수 있음 return data; }
다음은 코드 전문이다.
import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.util.Comparator; import java.util.PriorityQueue; import java.util.StringTokenizer; public class Main { static int[] xmove = {-1,0,0,1}; static int[] ymove = {0,-1,1,0}; public static int move(int size, int howManyEat, int[][] map) { int time = 0; int x = -1; //상어 위치 x int y = -1; //상어 위치 y for(int i=0; i<map.length; i++) { for(int j=0; j<map.length; j++) { if(map[i][j] == 9) { x = i; y = j; } } } int[] list = bfs(map, size); //먹을 수 있는 물고기가 없으면 끝 if(list[0] == -1 || list[1] == -1 || list[2] == -1) { //먹을 수 없음 return time; }else { //먹을 수 있음 howManyEat = howManyEat+1; time += list[0]; if(size == howManyEat) { size++; howManyEat = 0; } //System.out.println("먹으려는 애 좌표:"+list[1]+","+list[2]); map[x][y] = 0; map[list[1]][list[2]] = 9; time += move(size, howManyEat, map); } return time; } //다음에 먹을 먹이 위치와 걸린 시간 반환 public static int[] bfs(int[][] map, int size) { int[] ans = new int[3]; //초기화 ans[0] = -1; ans[1] = -1; ans[2] = -1; int x = -1; //상어 위치 x int y = -1; //상어 위치 y int[][] graph = new int[map.length][map[0].length]; //몇칸 이동했는지 표시하는 배열 loop: for(int i=0; i<map.length; i++) { for(int j=0; j<map.length; j++) { if(map[i][j] == 9) { x = i; y = j; break loop; } } } PriorityQueue<int[]> queue = new PriorityQueue<int[]>(new Comparator<int[]>() { @Override public int compare(int[] a, int[] b) { //시간이 같을 때 if(Integer.compare(a[0],b[0])==0) { //x좌표 같을 때 if(Integer.compare(a[1],b[1])==0) { //y좌표 같을 때 return Integer.compare(a[2],b[2]); } return Integer.compare(a[1],b[1]); } return Integer.compare(a[0],b[0]); } }); queue.offer(new int[] {0, x, y}); while(!queue.isEmpty()) { int[] data = queue.poll(); int cur_i = data[1]; int cur_j = data[2]; if(map[cur_i][cur_j]>0 && map[cur_i][cur_j] < size && map[cur_i][cur_j] != 9) { //먹을 수 있음 return data; } for(int i=0; i<4; i++) { int next_i = cur_i + xmove[i]; int next_j = cur_j + ymove[i]; if(next_i<0 || next_i>=map.length || next_j<0 || next_j>=map[0].length || map[next_i][next_j] > size || graph[next_i][next_j] != 0) { continue; } //이동 가능 graph[next_i][next_j] = graph[cur_i][cur_j] + 1; queue.offer(new int[] {graph[next_i][next_j], next_i, next_j}); } } return ans; } public static void main(String[] args) throws IOException { BufferedReader br = new BufferedReader(new InputStreamReader(System.in)); StringTokenizer st = new StringTokenizer(br.readLine()); int n = Integer.parseInt(st.nextToken()); int[][] map = new int[n][n]; int size = 2; //아기상어 크기 int howManyEat = 0; //먹은 먹이 개수 for(int i=0; i<n; i++) { st = new StringTokenizer(br.readLine()); for(int j=0; j<n; j++) { int k = Integer.parseInt(st.nextToken()); map[i][j] = k; } } System.out.println(move(size, howManyEat, map)); } }
테스트 케이스는 다음을 참고하자.
4 4 3 2 1 0 0 0 0 0 0 9 0 1 2 3 4 14 //위쪽,왼쪽이 우선순위가 높게 하기 6 5 4 3 2 3 4 4 3 2 3 4 5 3 2 9 5 6 6 2 1 2 3 4 5 3 2 1 6 5 4 6 6 6 6 6 6 60 //내가 나를 먹으면 안 됨 7 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 9 48
'코테 풀이' 카테고리의 다른 글
백준 풀이 : 1806 부분합 (2) 2024.10.09 백준 15686 치킨 배달 - 조합, 백트레킹 (1) 2024.02.07 백준 7562번: 나이트의 이동 (1) 2023.12.18 백준 2579번: 계단 오르기 (1) 2023.12.17 백준 22862번: 가장 긴 짝수 연속한 부분 수열(large) (0) 2023.12.04