행렬을 리스트로 변환하여 풀이하면 접근하기 쉬울 때가 있다.
matrixToList() : 행렬을 리스트로 변환.listToMatrix() : 리스트를 행렬로 변환.dirChange() : 방향 변환. (앞에 두 메소드에 사용)blizzard() : 구슬 파괴.boom() : 구슬 파괴 후 연속적인 폭발.change() : 폭발이 끝난 후 구슬들을 변환.blizzard()를 실행하여 범위안에 인덱스를 0으로 만든다. (구슬 파괴)matrixToList()를 실행하여 행렬인 구슬 정보들을 리스트로 변환한다.boom()을 실행하여 폭파할 구슬이 없을 때 까지 4개 이상의 연속적인 구슬들을 폭파시키고 얻은 점수를 answer에 더한다.change()를 실행하여 boom()까지 진행 된 리스트를 조건에 맞게 변화시킨다.listToMatrix()를 실행하여 블리자드가 진행 된 리스트를 다시 행렬로 변환한다.import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
public class Main {
    static final int UP = 1;
    static final int DOWN = 2;
    static final int LEFT = 3;
    static final int RIGHT = 4;
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        int[] size = Arrays.stream(br.readLine().split(" "))
            .mapToInt(Integer::parseInt)
            .toArray();
        int N = size[0];
        int M = size[1];
        int userIdx = N / 2;
        int[][] matrix = new int[N][N];
        for (int i = 0; i < N; i++) {
            int[] row = Arrays.stream(br.readLine().split(" "))
                .mapToInt(Integer::parseInt)
                .toArray();
            matrix[i] = row;
        }
        matrix[userIdx][userIdx] = -1;
        int[][] blizzards = new int[M][2];
        for (int i = 0; i < M; i++) {
            int[] blizzard = Arrays.stream(br.readLine().split(" "))
                .mapToInt(Integer::parseInt)
                .toArray();
            blizzards[i] = blizzard;
        }
        int answer = 0;
        for (int[] blizzard : blizzards) {
            blizzard(matrix, userIdx, blizzard[0], blizzard[1]);
            List<Integer> list = matrixToList(matrix, userIdx);
            answer += boom(list);
            list = change(N, list);
            matrix = listToMatrix(list, N, userIdx);
        }
        System.out.println(answer);
    }
    public static List<Integer> matrixToList(int[][] matrix, int userIdx) {
        List<Integer> list = new ArrayList<>();
        int curX = userIdx;
        int curY = userIdx;
        int dir = LEFT;
        int length = 1;
        int move = 0;
        int corner = 0;
        while (curX != 0 || curY != -1) {
            list.add(matrix[curX][curY]);
            if (move == length) {
                dir = dirChange(dir);
                corner++;
                move = 0;
            }
            if (corner == 2) {
                length++;
                corner = 0;
            }
            switch (dir) {
                case UP:
                    curX--;
                    break;
                case DOWN:
                    curX++;
                    break;
                case LEFT:
                    curY--;
                    break;
                default:
                    curY++;
                    break;
            }
            move++;
        }
        list.removeAll(Collections.singletonList(0));
        return list;
    }
    public static int[][] listToMatrix(List<Integer> list, int N, int userIdx) {
        int[][] matrix = new int[N][N];
        int curX = userIdx;
        int curY = userIdx;
        int dir = LEFT;
        int length = 1;
        int move = 0;
        int corner = 0;
        for (int i = 0; i < list.size(); i++) {
            matrix[curX][curY] = list.get(i);
            if (move == length) {
                dir = dirChange(dir);
                corner++;
                move = 0;
            }
            if (corner == 2) {
                length++;
                corner = 0;
            }
            switch (dir) {
                case UP:
                    curX--;
                    break;
                case DOWN:
                    curX++;
                    break;
                case LEFT:
                    curY--;
                    break;
                default:
                    curY++;
                    break;
            }
            move++;
        }
        return matrix;
    }
    
    public static int dirChange(int dir) {
        switch (dir) {
            case UP:
                return LEFT;
            case DOWN:
                return RIGHT;
            case LEFT:
                return DOWN;
            default:
                return UP;
        }
    }
    public static void blizzard(int[][] matrix, int userIdx, int dir, int range) {
        for (int i = 1; i <= range; i++) {
            switch (dir) {
                case UP:
                    matrix[userIdx - i][userIdx] = 0;
                    break;
                case DOWN:
                    matrix[userIdx + i][userIdx] = 0;
                    break;
                case LEFT:
                    matrix[userIdx][userIdx - i] = 0;
                    break;
                default:
                    matrix[userIdx][userIdx + i] = 0;
                    break;
            }
        }
    }
    public static int boom(List<Integer> list) {
        int totalScore = 0;
        while (true) {
            int score = 0;
            int same = 1;
            int lastNum = 0;
            int curNum;
            for (int i = list.size() - 1; i >= 0; i--) {
                curNum = list.get(i);
                if (curNum == lastNum) {
                    same++;
                } else {
                    if (same >= 4) {
                        score += lastNum * same;
                        list.subList(i + 1, i + same + 1).clear();
                    }
                    same = 1;
                }
                lastNum = curNum;
            }
            if (score == 0) {
                break;
            }
            totalScore += score;
        }
        return totalScore;
    }
    public static List<Integer> change(int N, List<Integer> list) {
        int curNum;
        int lastNum = list.get(list.size() - 1);
        int same = 1;
        for (int i = list.size() - 2; i >= 0; i--) {
            curNum = list.get(i);
            if (curNum == lastNum) {
                same++;
            } else {
                if (same == 1) {
                    list.add(i + 2, lastNum);
                    list.set(i + 1, same);
                } else if (same == 2) {
                    list.set(i + 2, lastNum);
                    list.set(i + 1, same);
                } else {
                    list.remove(i + 3);
                    list.set(i + 2, lastNum);
                    list.set(i + 1, same);
                }
                same = 1;
            }
            lastNum = curNum;
        }
        int maxSize = N * N;
        return list.size() > maxSize ? new ArrayList<>(list.subList(0, maxSize)) : list;
    }
}