[백준] 1938: 통나무 옮기기 (Java)

Yoon Uk·2022년 8월 3일
0

알고리즘 - 백준

목록 보기
45/130
post-thumbnail
post-custom-banner

문제

BOJ 1938: 통나무 옮기기 https://www.acmicpc.net/problem/1938

풀이

조건

  • 통나무를 움직이는 방법은 , , , , 90도 회전이 있다.
  • 회전이 가능하기 위해서는 통나무를 둘러싸고 있는 3*3 정사각형 구역에 나무가 하나도 없어야 한다.

풀이 순서

  • 통나무의 시작 좌표도착 위치의 좌표를 각각 저장해 둔다.
  • bfs를 사용하여 최소 동작 횟수를 구한다.
    • 3차원 배열을 사용하여 통나무의 방향에 따라 방문 처리를 따로 해준다.
    • 시작 통나무의 방향을 구한다.
    • Queue시작 통나무 중간 위치의 좌표, 방향, 동작 횟수(0)를 넣는다.
    • while문을 돌며 탐색한다.
      • 통나무의 중간이 목적지의 중간에 도달하면 방향에 맞게 도달했는지 확인하고 종료한다.
      • 상하좌우를 탐색하며 통나무를 옮길 수 있다면 Queue에 넣어준다.
      • 통나무를 회전할 수 있다면 Queue에 넣어준다.

코드

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

public class Main {
    
	static int N;
	static char[][] map;
	
	static Point[] SW, EW;
	
	static int[] dx = {-1, 1, 0, 0};
	static int[] dy = {0, 0, -1, 1};
    
    public static void main(String[] args) throws IOException {
    	BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    	
    	N = Integer.parseInt(br.readLine());
    	map = new char[N][N];
    	
    	SW = new Point[3]; // 시작 통나무 위치
    	EW = new Point[3]; // 도착 위치
    	
    	int sIdx = 0, eIdx = 0;
    	for(int i=0; i<N; i++) {
    		String str = br.readLine();
    		for(int j=0; j<N; j++) {
    			map[i][j] = str.charAt(j);
    			
    			if(map[i][j] == 'B') {
    				SW[sIdx++] = new Point(i, j);
    			}
    			if(map[i][j] == 'E') {
    				EW[eIdx++] = new Point(i, j);
    			}
    		}
    	}
        
    	int ans = bfs();
    	
    	System.out.println(ans);
    	
    }
    
    static int bfs() {
    	Queue<Wood> que = new LinkedList<>();
    	boolean[][][] isVisited = new boolean[2][N][N]; // 가로인 경우, 세로인 경우 나눠서 방문 체크
    	
    	// 통나무 상태 확인
    	int dir = 0;
    	// 통나무가 가로 방향 이라면
    	if(SW[0].y + 1 == SW[1].y) {
    		dir = 0; // 가로 방향
    	} else {
    		dir = 1; // 세로 방향
    	}
    	
    	// 통나무의 중간을 기준으로 이동
    	que.add(new Wood(SW[1].x, SW[1].y, dir, 0));
    	isVisited[dir][SW[1].x][SW[1].y] = true;
    	
    	while(!que.isEmpty()) {
    		
    		Wood w = que.poll();
    		int curX = w.x;
    		int curY = w.y;
    		int curDir = w.dir;
    		int curDist = w.dist;
    		
    		// 통나무의 중간이 목적지의 중간에 도달하면
    		if(curX == EW[1].x && curY == EW[1].y) {
    			// 통나무가 가로방향이면
    			if(curDir == 0 && map[curX][curY - 1] == 'E' && map[curX][curY + 1] == 'E') {
    				return curDist;
    			} 
    			// 통나무가 세로방향이면
    			else if(curDir == 1 && map[curX - 1][curY] == 'E' && map[curX + 1][curY] == 'E') {
    				return curDist;
    			}
    		}
    		
    		// 상하좌우 탐색
    		for(int t=0; t<4; t++) {
    			int nx = curX + dx[t];
    			int ny = curY + dy[t];
    			
    			// 통나무가 이동할 수 있는지 확인
    			if(!checkRange(curDir, nx, ny, t)) continue;
    			
    			// 방문 여부 체크
    			if(isVisited[curDir][nx][ny]) continue;
    			
    			isVisited[curDir][nx][ny] = true;
    			que.add(new Wood(nx, ny, curDir, curDist + 1));
    		}
    		
    		// 회전 가능한 지 체크
    		if(canRotate(curX, curY)) {
    			// 통나무가 가로일 때
    			if(curDir == 0 && !isVisited[1][curX][curY]) {
    				isVisited[1][curX][curY] = true;
    				que.add(new Wood(curX, curY, 1, curDist + 1)); // 방향 바꿔서 넣어줌
    			} 
    			// 통나무가 세로일 때
    			else if(curDir == 1 && !isVisited[0][curX][curY]) {
    				isVisited[0][curX][curY] = true;
    				que.add(new Wood(curX, curY, 0, curDist + 1)); // 방향 바꿔서 넣어줌
    			}
    		}
    		
    	}
    	  	
    	return 0;
    }
    
    // 옮길 수 있는 지 확인하는 메소드
    static boolean checkRange(int dir, int x, int y, int t) {
    	
    	switch(dir) {
    		// 가로 방향일 때 수평 방향으로 이동할 수 있는지 확인
	    	case 0:
	    		// 상/하로 이동
	    		if(t < 2) {
	    			// 통나무 끝이 범위를 넘어가면 false
	    			if(x < 0 || x >= N) return false;
	    			// 나무가 있을 경우 false
	    			if(map[x][y] == '1' || map[x][y - 1] == '1' || map[x][y + 1] == '1') return false;
	    		} 
	    		// 좌/우로 이동
	    		else {
	    			// 통나무 끝이 범위를 넘어가면 false
	    			if(y - 1 < 0 || y + 1 >= N) return false;
	    			// 나무가 있을 경우 false
	    			if(map[x][y] == '1' || map[x][y - 1] == '1' || map[x][y + 1] == '1') return false;
	    		}

	    		break;
	    	// 세로 방향일 때 수직 방향으로 이동할 수 있는지 확인	
	    	case 1:
	    		// 상/하로 이동
	    		if(t < 2) {
	    			// 통나무 끝이 범위를 넘어가면 false
	    			if(x - 1 < 0 || x + 1 >= N) return false;
	    			// 나무가 있을 경우 false
	    			if(map[x][y] == '1' || map[x - 1][y] == '1' || map[x + 1][y] == '1') return false;
	    		} 
	    		// 좌/우로 이동
	    		else {
	    			// 통나무 끝이 범위를 넘어가면 false
	    			if(y < 0 || y >= N) return false;
	    			// 나무가 있을 경우 false
	    			if(map[x][y] == '1' || map[x - 1][y] == '1' || map[x + 1][y] == '1') return false;
	    		}
	    		
	    		break;
	    		
	    	default:
	    		break;
    	}

    	return true;
    }
    
    // 회전시킬 수 있는지 확인하는 메소드
    static boolean canRotate(int x, int y) {
    	
    	for(int i = x - 1; i <= x + 1; i++) {
    		for(int j = y - 1; j <= y + 1; j++) {
    			// 범위 나간 경우
    			if(i < 0 || i >= N || j < 0 || j >= N) {
    				return false;
    			}
    			// 나무 있는 경우
    			if(map[i][j] == '1') {
    				return false;
    			}
    		}
    	}
    	
    	return true;
    }
    
    // 통나무 중간의 위치, 방향, 굴린 횟수 등 넣을 구조체
    static class Wood{
    	int x, y, dir, dist;
    	
    	Wood(int x, int y, int dir, int dist){
    		this.x = x;
    		this.y = y;
    		this.dir = dir;
    		this.dist = dist;
    	}
    }
    
    // 시작 지점과 종료 지점의 위치를 넣을 구조체
    static class Point{
    	int x, y;
    	
    	Point(int x, int y){
    		this.x = x;
    		this.y = y;
    	}
    }

}

정리

  • 통나무를 회전시키는 데 방향값(dir 변수)만 사용하면 쉽다는 것을 배웠다.
  • 처음부터 동작 계획 등 설계를 꼼꼼히 해야 시간 낭비를 최소화 할 수 있다.
post-custom-banner

0개의 댓글