728x90
안녕하세요 Coding-Knowjam입니다.
오늘은 백준 온라인 저지에 있는 12100번 2048(easy) 문제를 Java로 풀어보겠습니다.
문제에 대한 링크는 아래에 있으니 문제를 먼저 읽고 와주시길 바랍니다.
https://www.acmicpc.net/problem/12100
1. 문제 설명
해당 문제는 구현 문제로서 주어진 내용에 알맞게 구현만 하면 풀 수 있습니다.
문제에서 주어진 제약조건은 다음과 같습니다.
- 한 번에 한 방향으로만 이동이 가능하며 이동할 때 모든 블록이 같이 움직임
- 이동하는 과정에서 같은 값을 가진 블록을 만나면 이동방향에 있는 블록의 위치로 값이 합쳐짐
- 이동하는 과정에서 한번 합쳐진 적이 있던 블록은 다시 같은 값을 만나도 합쳐질 수 없음
위의 3개의 조건을 고려해서 코드로 구현하면 되며, 보드의 크기가 최대 20X20 사이즈이므로 완전 탐색을 통해서 모든 경우의 수를 체크해 볼 수 있습니다.
저는 완전 탐색을 위해 재귀 형태로 구현을 하였으며, 코드는 아래와 같습니다.
2. Java로 문제 풀이
package study.blog.codingnojam.algorithm.boj;
import java.io.*;
// 백준온라인저지 12100 2048(easy) 문제 풀이 Java
public class BOJ_12100 {
// 상하좌우 이동을 반복문으로 구현하기 위해 사용할 배열
static int[] moveR = {-1, 1, 0, 0};
static int[] moveC = {0, 0, -1, 1};
// 결과를 저장할 변수
static int result = 0;
public static void main(String[] args) throws IOException {
// 입력을 받을 객체
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
// 보드 크기
int n = Integer.parseInt(br.readLine());
// 보드
int[][] board = new int[n][n];
// 보드판 정보 받아서 초기화
for (int i = 0; i < n; i++) {
String[] info = br.readLine().split(" ");
for (int j = 0; j < n; j++) {
board[i][j] = Integer.parseInt(info[j]);
}
}
// 5번 이하로 움직이는 모든 경우를 판달할 재귀 메서드
recursion(0, 5, board);
// 결과 출력
System.out.println(result);
}
/**
* 이동하는 모든 경우의 수를 체크할 재귀메서드
* @param index : 재귀메서드 진행 횟수
* @param limit : 보드가 움직여야하는 횟수
* @param board : 보드판
*/
static void recursion(int index, int limit, int[][] board) {
// 보드가 5번 움직였을 겨우
if (index == limit) {
// 각각의 경우의 수마다 결과 값을 저장할 임시변수
int tempReulst = 0;
// 보드판의 모든 값을 순회해서 가장 높은 숫자를 결과 임시변수에 저장
for (int i = 0; i < board.length; i++) {
for (int j = 0; j < board.length; j++) {
tempReulst = Math.max(tempReulst, board[i][j]);
}
}
// 임시결과변수와 결과변수를 비교해서 큰 값을 결과변수에 저장 후 종료
result = Math.max(result, tempReulst);
return;
}
// 반복문으로 상하좌우 이동
for (int i = 0; i < 4; i++) {
// 보드 내에서 숫자가 합쳐졌는지 여부를 판단할 배열
boolean[][] sumChk = new boolean[board.length][board.length];
// 각각의 경우의 수마다 보드판을 조작해야 하므로 임시보드판 생성 후 초기화
int[][] tempBoard = new int[board.length][board.length];
for (int q = 0; q < board.length; q++) {
for (int j = 0; j < board.length; j++) {
tempBoard[q][j] = board[q][j];
}
}
// 보드판의 값들을 위로, 왼쪽으로 움직이는 경우
if (i == 0 || i==2) {
// 보드판의 좌측 상단부터 시작
for (int j = 0; j < tempBoard.length; j++) {
for (int k = 0; k < tempBoard[j].length; k++) {
// 정해진 방향으로 보드판의 값들을 움직이기
move(tempBoard, sumChk, i, j, k);
}
}
}
// 보드판의 값들을 아래로, 오른쪽으로 움직이는 경우
else if (i == 1 || i == 3) {
// 보드판의 우측 하단부터 시작
for (int j = tempBoard.length - 1; j >= 0; j--) {
for (int k = tempBoard[j].length; k >= 0; k--) {
// 정해진 방향으로 보드판의 값들을 움직이기
move(tempBoard, sumChk, i, j, k);
}
}
}
// 한번의 이동이 완료된 후 재귀 호출
recursion(index +1, limit, tempBoard);
}
}
/**
* 보드판의 값들을 움직이기 위해 사용하는 메서드
* @param board : 보드판을 표현할 배열
* @param sumChk : 보드판의 값들이 합쳐졌는지 여부를 체크하기 위한 배열
* @param i : 어느 방향으로 움직이는지를 구분하기 위한 변수 (0:상, 1:하, 2:좌, 3:우)
* @param j : 시작 행
* @param k : 시작 열
*/
private static void move(int[][] board, boolean[][] sumChk, int i, int j, int k) {
// 보드판의 값 체크를 시작할 행과 열 받기
int nowRow = j;
int nowCol = k;
// 이동할 위치의 행과 열 계산
int moveRow = nowRow + moveR[i];
int moveCol = nowCol + moveC[i];
// 계산된 위치가 보드판을 벗어나면 종료
if (moveRow >= board.length || moveRow < 0 || moveCol >= board.length || moveCol < 0) {
return;
}
// 이동의 종료 여부를 판단할 변수
boolean end = false;
// 이동 시작
while (!end) {
// 이동할 위치에 값이 0이면
if (board[moveRow][moveCol] == 0) {
// 현재 위치의 값을 이동할 위치의 값으로 저장
board[moveRow][moveCol] = board[nowRow][nowCol];
// 현재 위치의 값을 0으로 변경
board[nowRow][nowCol] = 0;
// 현재 위치의 행과열 값 갱신
nowRow = moveRow;
nowCol = moveCol;
// 다음에 이동할 위치 계산
moveRow = nowRow + moveR[i];
moveCol = nowCol + moveC[i];
// 계산된 위치가 보드판을 벗어나면 이동 종료
if (moveRow >= board.length || moveRow < 0 || moveCol >= board.length || moveCol < 0) {
end = true;
}
}
// 이동할 위치에 값이 현재의 값과 동일하면
else if (board[moveRow][moveCol] == board[nowRow][nowCol]) {
// 이동할 위치의 값이 합쳐진 이력이 없을 경우 블록을 합치고 종료
if (!sumChk[moveRow][moveCol]) {
// 이동할 위치의 값 갱신 (블록 2개가 합쳐지는 것)
board[moveRow][moveCol] = board[moveRow][moveCol] * 2;
// 현재 위치의 값 0으로 변경
board[nowRow][nowCol] = 0;
// 이동할 위치의 값이 합쳐진 적이 있다는 이력 남기기
sumChk[moveRow][moveCol] = true;
}
// 이동할 위치의 값이 합쳐진 이력이 존재하면 아무것도 하지않고 이동 종료
end = true;
}
// 이동할 위치의 값이 현재 위치의 값과 다르다면 이동 종료
else {
end = true;
}
}
}
}
작성한 코드의 라인마다 주석을 달아놓았으니, 주석과 코드를 같이 살펴보시면 이해하는데 크게 어렵지 않으실 겁니다.
혹시나 이해가 가지 않는 부분이 있으시다면 댓글을 남겨주시길 바랍니다.
감사합니다.
728x90
'Algorithm & Data Structure > 문제풀이' 카테고리의 다른 글
[Algorithm] 백준 17140(BOJ 17140) 이차원 배열과 연산 문제풀이!! (Java) (0) | 2021.08.28 |
---|---|
[Algorithm] 백준 3190번(BOJ 3190) 뱀 문제풀이!! (java) (0) | 2021.08.25 |
[Algorithm] 백준 4673(BOJ 4673) 셀프 넘버 문제풀이 (Java) (0) | 2021.08.10 |
[Algorithm] 백준 9012(BOJ 9012) 괄호 문제풀이 (Java) (0) | 2021.08.10 |
[Algorithm] 백준 9095번 (BOJ 9095) 1,2,3 더하기 문제풀이 (Java) (0) | 2021.08.09 |
댓글