알고리즘/[ Baekjoon ]

[ BOJ ][JAVA][2615] 오목

kim.svadoz 2021. 4. 24. 01:18
반응형

www.acmicpc.net/problem/2615

 

2615번: 오목

오목은 바둑판에 검은 바둑알과 흰 바둑알을 교대로 놓아서 겨루는 게임이다. 바둑판에는 19개의 가로줄과 19개의 세로줄이 그려져 있는데 가로줄은 위에서부터 아래로 1번, 2번, ... ,19번의 번호

www.acmicpc.net

시간 제한 메모리 제한 제출 정답 맞은 사람 정답 비율
1 초 128 MB 9066 1842 1438 21.402%

문제

오목은 바둑판에 검은 바둑알과 흰 바둑알을 교대로 놓아서 겨루는 게임이다. 바둑판에는 19개의 가로줄과 19개의 세로줄이 그려져 있는데 가로줄은 위에서부터 아래로 1번, 2번, ... ,19번의 번호가 붙고 세로줄은 왼쪽에서부터 오른쪽으로 1번, 2번, ... 19번의 번호가 붙는다.

img

위의 그림에서와 같이 같은 색의 바둑알이 연속적으로 다섯 알을 놓이면 그 색이 이기게 된다. 여기서 연속적이란 가로, 세로 또는 대각선 방향 모두를 뜻한다. 즉, 위의 그림은 검은색이 이긴 경우이다. 하지만 여섯 알 이상이 연속적으로 놓인 경우에는 이긴 것이 아니다.

입력으로 바둑판의 어떤 상태가 주어졌을 때, 검은색이 이겼는지, 흰색이 이겼는지 또는 아직 승부가 결정되지 않았는지를 판단하는 프로그램을 작성하시오. 단, 검은색과 흰색이 동시에 이기거나 검은색 또는 흰색이 두 군데 이상에서 동시에 이기는 경우는 입력으로 들어오지 않는다.

입력

19줄에 각 줄마다 19개의 숫자로 표현되는데, 검은 바둑알은 1, 흰 바둑알은 2, 알이 놓이지 않는 자리는 0으로 표시되며, 숫자는 한 칸씩 띄어서 표시된다.

출력

첫줄에 검은색이 이겼을 경우에는 1을, 흰색이 이겼을 경우에는 2를, 아직 승부가 결정되지 않았을 경우에는 0을 출력한다. 검은색 또는 흰색이 이겼을 경우에는 둘째 줄에 연속된 다섯 개의 바둑알 중에서 가장 왼쪽에 있는 바둑알(연속된 다섯 개의 바둑알이 세로로 놓인 경우, 그 중 가장 위에 있는 것)의 가로줄 번호와, 세로줄 번호를 순서대로 출력한다.

예제 입력 1

0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 1 2 0 0 2 2 2 1 0 0 0 0 0 0 0 0 0 0
0 0 1 2 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0
0 0 0 1 2 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 1 2 2 0 0 0 0 0 0 0 0 0 0 0 0
0 0 1 1 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 2 1 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0

예제 출력 1

1
3 2

코드

/*
    오목!!
    6개 이상인 것들을 체크하는 부분이 까다로웠다.
    따라서 각 방향(4개)에 따른 3차원 배열을 생성해서 해결한다.
    // 가로(->) 세로(아래), 대각선 2개(오른쪽 아래, 오른쪽 위)
    static int[] dx = { 1, 1, 0, -1 };
    static int[] dy = { 0, 1, 1, 1 };
*/
import java.io.*;
import java.util.*;
public class p2615 {
    static int map[][] = new int[21][21];
    static int[][][] memo = new int[21][21][4];
    static int dr[] = {1, 1, 0, -1};
    static int dc[] = {0, 1, 1, 1};
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;

        for (int i = 1; i <= 19; i++) {
            st = new StringTokenizer(br.readLine());
            for (int j = 1; j <= 19; j++) {
                map[i][j] = Integer.parseInt(st.nextToken());
            }
        }
        System.out.println(solve());
    }
    static String solve() {
        for (int i = 1; i <= 19; i++) {
            for (int j = 1; j <= 19; j++) {
                if (map[j][i] != 0) { // 여기서 map[i][j]가 아니라 왜 map[j][i]인지 고민해보자.
                    for (int d = 0; d < 4; d++) {
                        if (memo[j][i][d] == 0 && calc(j, i, d, map[j][i]) == 5) {
                            return map[j][i] + "\n" + j + " " + i;
                        }
                    }
                }
            }
        }
        return "0";
    }
    static int calc(int r, int c, int dir, int color) {
        int nr = r + dr[dir];
        int nc = c + dc[dir];

        if (map[nr][nc] == color) {
            return memo[nr][nc][dir] = calc(nr, nc, dir, color) + 1;
        }
        return 1;
    }
}

BFS 풀이

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.Queue;
import java.util.LinkedList;
public class Main {
    static boolean[][] visited = new boolean[19][19];
    static char[][] map = new char[19][19];
    static int[] dx = {1, 0, 1, -1};
    static int[] dy = {1, 1, 0, 1};
    public static void main(String[] args) throws Exception{
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        for(int i=0; i<19; i++) {
            String input = br.readLine().replace(" ", "");
            for(int j=0; j<19; j++) {
                map[i][j] = input.charAt(j);
            }
        }
        for(int i=0; i<19; i++) {
            for(int j=0; j<19; j++) {
                if(map[i][j]!='0') {
                    for(int d=0; d<4; d++) {
                        int cnt = bfs(i, j, dk);
                        if(cnt==5) {
                            System.out.println(map[i][j]);
                            System.out.println((i+1)+" "+(j+1));
                            return;
                        }
                    }
                }
            }
        }
        System.out.println(0);
    }
    public static int bfs(int x, int y, int dir) {
        Queue<Pair> queue = new LinkedList<>();
        queue.add(new Pair(x, y, 1));
        int max = 0;
        while(!queue.isEmpty()) {
            Pair temp = queue.poll();
            max = Math.max(temp.cnt, max);
            int nx = temp.x+dx[dir];
            int ny = temp.y+dy[dir];
            if(nx<0 || nx>=19 || ny<0 || ny>=19 || map[nx][ny]!=map[temp.x][temp.y]) continue;
            queue.add(new Pair(nx, ny, temp.cnt+1));
        }
        if(max==5) {
            int nx = x-dx[idx];
            int ny = y-dy[idx];
            if(nx>=0 && nx<19 && ny>=0 && ny<19 && map[nx][ny]==map[x][y])
                max++;
        }
        return max;
    }
    public static class Pair {
        int x;
        int y;
        int cnt;
        public Pair(int x, int y, int cnt) {
            this.x = x;
            this.y = y;
            this.cnt = cnt;
        }
    }
}
반응형