알고리즘

백준 - 14890 경사로(Java)

ksb-dev 2023. 3. 20. 15:51

https://www.acmicpc.net/problem/14890

 

14890번: 경사로

첫째 줄에 N (2 ≤ N ≤ 100)과 L (1 ≤ L ≤ N)이 주어진다. 둘째 줄부터 N개의 줄에 지도가 주어진다. 각 칸의 높이는 10보다 작거나 같은 자연수이다.

www.acmicpc.net

 

구현 문제입니다.

 

구현 순서는 다음과 같습니다.

  1. 이차원 배열에서 가로, 세로 행을 따로 추출한다.
  2. 추출된 배열의 높이가 같은지 확인한다.
  3. 높이가 같지 않다면 경사로를 겹치지 않게 설치할 수 있는지 확인한다
  4. 경사로를 겹치지 않게 설치할 수 있다면 설치한다.

 

[구현 코드]

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

public class Main {
    private static final String SEPARATOR = " ";

    private static int n, l;

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

        n = Integer.parseInt(stz.nextToken());
        l = Integer.parseInt(stz.nextToken());
        int answer = 0;

        int[][] map = new int[n][n];

        for (int i = 0; i < n; i++) {
            stz = new StringTokenizer(br.readLine(), SEPARATOR);
            for (int j = 0; j < n; j++) {
                map[i][j] = Integer.parseInt(stz.nextToken());
            }
        }

        // 가로행
        for (int i = 0; i < n; i++) {
            int[] forward = map[i];
            answer += getRoadCount(forward);
        }

        // 세로행
        for (int j = 0; j < n; j++) {
            int[] forward = new int[n];
            for (int i = 0; i < n; i++) {
                forward[i] = map[i][j];
            }
            answer += getRoadCount(forward);
        }

        bw.write(String.valueOf(answer));

        bw.flush();
        bw.close();
        br.close();
    }

    private static int getRoadCount(int[] forward) {
        if (isRoad(forward)) return 1;
        return 0;
    }

    private static boolean isRoad(int[] heights) {
        int h = heights[0];
        int len = 1;
        boolean[] incline = new boolean[n]; // 경사면이 설치유무

        for (int i = 1; i < n; i++) {
            if (heights[i] == h) { // 높이가 같다면
                len++;
            } else if (heights[i] - h == 1) { // 올라갈 때
                if (len >= l) {
                    for (int j = i - l; j < i; j++) {
                        if (incline[j]) return false; // 새로 설치하는 경사면에 다른 경사면이 겹치게 되면
                    }
                    h = heights[i];
                    len = 1;
                } else {
                    return false;
                }
            } else if (h - heights[i] == 1) { // 내려갈 때
                if (i + (l - 1) < n) {
                    for (int j = i + 1; j < i + (l); j++) {
                        if (heights[i] != heights[j] || incline[j]) { // 경사면을 설치가능 유무 확인
                            return false;
                        }
                    }
                    // 경사면 설치
                    Arrays.fill(incline, i + 1, i + l, true);
                    if (l == 1) incline[i] = true;

                    // 경사면 설치이 이후로 위치 조정
                    h = heights[i];
                    len = 1;
                    i += (l - 1);
                } else {
                    return false;
                }
            } else { // 높이 차이가 2 이상일 때
                return false;
            }
        }
        return true;
    }
}