[백준] 20057 - 마법사 상어와 토네이도 (java)

HaYeong Jang·2021년 2월 17일
0

[백준] 알고리즘

목록 보기
45/62
post-thumbnail

문제

마법사 상어가 토네이도를 배웠고, 오늘은 토네이도를 크기가 N×N인 격자로 나누어진 모래밭에서 연습하려고 한다. 위치 (r, c)는 격자의 r행 c열을 의미하고, A[r][c]는 (r, c)에 있는 모래의 양을 의미한다.

토네이도를 시전하면 격자의 가운데 칸부터 토네이도의 이동이 시작된다. 토네이도는 한 번에 한 칸 이동한다. 다음은 N = 7인 경우 토네이도의 이동이다.

토네이도가 한 칸 이동할 때마다 모래는 다음과 같이 일정한 비율로 흩날리게 된다.

토네이도가 x에서 y로 이동하면, y의 모든 모래가 비율과 α가 적혀있는 칸으로 이동한다. 비율이 적혀있는 칸으로 이동하는 모래의 양은 y에 있는 모래의 해당 비율만큼이고, 계산에서 소수점 아래는 버린다. α로 이동하는 모래의 양은 비율이 적혀있는 칸으로 이동하지 않은 남은 모래의 양과 같다. 모래가 이미 있는 칸으로 모래가 이동하면, 모래의 양은 더해진다. 위의 그림은 토네이도가 왼쪽으로 이동할 때이고, 다른 방향으로 이동하는 경우는 위의 그림을 해당 방향으로 회전하면 된다.

토네이도는 (1, 1)까지 이동한 뒤 소멸한다. 모래가 격자의 밖으로 이동할 수도 있다. 토네이도가 소멸되었을 때, 격자의 밖으로 나간 모래의 양을 구해보자.

입력

첫째 줄에 격자의 크기 N이 주어진다. 둘째 줄부터 N개의 줄에는 격자의 각 칸에 있는 모래가 주어진다. r번째 줄에서 c번째 주어지는 정수는 A[r][c] 이다.

출력

격자의 밖으로 나간 모래의 양을 출력한다.

제한

  • 3 ≤ N ≤ 499
  • N은 홀수
  • 0 ≤ A[r][c] ≤ 1,000
  • 가운데 칸에 있는 모래의 양은 0

예제 입력

5
0 0 0 0 0
0 0 0 0 0
0 10 0 0 0
0 0 0 0 0
0 0 0 0 0

예제 출력

10

코드

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

public class Main {
    public static int[][] dr = {{-1, 1, -2, 2, -1, 1, -1, 1, 0, 0}, {-1, -1, 0, 0, 0, 0, 1, 1, 2, 1},
            {-1, 1, -2, 2, -1, 1, -1, 1, 0, 0}, {1, 1, 0, 0, 0, 0, -1, -1, -2, -1}}; // 서, 남, 동, 북 방향
    public static int[][] dc = {{1, 1, 0, 0, 0, 0, -1, -1, -2, -1}, {-1, 1, -2, 2, -1, 1, -1, 1, 0, 0},
            {-1, -1, 0, 0, 0, 0, 1, 1, 2, 1}, {-1, 1, -2, 2, -1, 1, -1, 1, 0, 0}};
    public static int[] per = {1, 1, 2, 2, 7, 7, 10, 10, 5};
    public static int[] next_r = {0, 1, 0, -1};
    public static int[] next_c = {-1, 0, 1, 0};
    public static int N;
    public static int[][] map;
    public static long ans = 0;

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

        N = Integer.parseInt(br.readLine());
        map = new int[N + 1][N + 1];

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

        tornado((N + 1) / 2, (N + 1) / 2, 0);

        bw.write(ans + "\n");
        br.close();
        bw.flush();
        bw.close();
    }

    public static void tornado(int row, int col, int d) {
        int cnt = 1;

        loop:
        for (int i = 1; i <= N; i++) {
            for (int j = 1; j <= 2; j++) {  // 두번씩 반복
                for (int l = 1; l <= cnt; l++) {
                    int nextRow = row + next_r[d];  // 이동할 위치
                    int nextCol = col + next_c[d];
                    int sum = 0;

                    if (nextRow == 1 && nextCol == 0) break loop;   // (1,1)에 위치했을 때 종료

                    int sand = map[nextRow][nextCol];   // 토네이도 한 칸 이동했을 때 모래의 양
                    for (int k = 0; k < 10; k++) {
                        if (k == 9) {   // 알파값 계산
                            if (nextRow + dr[d][k] < 1 || nextCol + dc[d][k] < 1 || nextRow + dr[d][k] > N || nextCol + dc[d][k] > N) {
                                ans += sand - sum;
                            } else {
                                map[nextRow + dr[d][k]][nextCol + dc[d][k]] += sand - sum;
                            }
                            break;
                        }

                        if (nextRow + dr[d][k] < 1 || nextCol + dc[d][k] < 1 || nextRow + dr[d][k] > N || nextCol + dc[d][k] > N) {
                            ans += sand * per[k] / 100;
                        } else {
                            map[nextRow + dr[d][k]][nextCol + dc[d][k]] += sand * per[k] / 100;
                        }
                        sum += sand * per[k] / 100;
                    }
                    row = nextRow;
                    col = nextCol;
                }
                d = (d + 1) % 4;    // 방향 바꾸기
            }
            cnt++;
        }
    }
}

정리

토네이도가 회전하는 것을 살펴보면,

1칸 -> 좌회전 -> 1칸 -> 좌회전
2칸 -> 좌회전 -> 2칸 -> 좌회전
.
.
(1,1)에 위치했을 때 종료

하는 것을 알 수 있다.
따라서 이동하는 칸 수를 cnt로 조절해 주고, 이동 -> 좌회전 을 2번씩 반복하도록 for 문을 만들어줬다.

알고리즘
1. (1,1)에 도달했을 때 가장 바깥 for 문을 종료시킨다.
2. 알파값인 경우와 비율이 적혀있는 경우를 구분하여 모래의 양을 계산해 준다.
3. map을 벗어나는 경우엔 ans에 모래의 양을 더해준다.
profile
기억하기 위해 기록하는 개발로그👣

0개의 댓글