Competition/Baekjoon

[백준] 2186번 자바 문자판

bisi 2020. 4. 24. 13:20
문제 출처 

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

 

2186번: 문자판

첫째 줄에 N(1 ≤ N ≤ 100), M(1 ≤ M ≤ 100), K(1 ≤ K ≤ 5)가 주어진다. 다음 N개의 줄에는 M개의 알파벳 대문자가 주어지는데, 이는 N×M 크기의 문자판을 나타낸다. 다음 줄에는 1자 이상 80자 이하의 영단어가 주어진다. 모든 문자들은 알파벳 대문자이며, 공백 없이 주어진다.

www.acmicpc.net

 

 

 

접근 방식 및 풀이

- DF + DFS 적용 문제이다. DFS 만으로 풀기에 시간이 너무 오래걸리므로 DF 알고리즘도 함께 적용한다.

- 참고 블로그

- 자세한 내용은 주석에다가 다 적어 놓았다.

 

 

 

소스 코드 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
 
public class Main {
 
    static int cnt = 0;
    static int N,M,K;
    static char arr[][];
    static int dp[][][];
    static boolean visit[][];
    static String target;
    static int moveX[] = {0,1,0,-1};
    static int moveY[] = {-1,0,1,0};
 
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st = new StringTokenizer(br.readLine());
 
        N = Integer.parseInt(st.nextToken()) ;
        M = Integer.parseInt(st.nextToken()) ;
        K = Integer.parseInt(st.nextToken()) ;
 
        arr = new char[N][M];
 
        for (int i = 0; i <N; i++) {
            String str = br.readLine();
            for (int j = 0; j <M ; j++) {
                arr[i][j] = str.charAt(j);
            }
        }
 
        target = br.readLine();
        dp = new int[N][M][target.length()];
 
        // 모두 -1 담아준다.
        for (int i = 0; i <N ; i++) {
            for (int j = 0; j <M ; j++) {
                for (int k = 0; k <target.length() ; k++) {
                    dp[i][j][k] = -1;
                }
            }
        }
 
 
        // dfs로 갈때마다 count 1 (새로운 단어를 만들어주는 경로로 가는것임.
        for (int i = 0; i <N ; i++) {
            for (int j = 0; j <M ; j++) {
                cnt += dfs(i,j,0);
            }
        }
        System.out.println(cnt);
    }
 
    // DFS 구현
    public static int dfs(int i, int j, int index){
        //DP에 알맞는 값을 채워 넣는다.
        // 1 방문했는데 가능성 있다. 2 방문했는데 가능성 없다 -1 방문하지 않았다.
        if(index ==target.length()-1//index 마지막에 왔다는 체크, count 하게 1로 리턴
            return dp[i][j][index]=1;
        if (dp[i][j][index]!=-1// -1이 아니면, 1아니면 2 이다 끝난 결과 값이므로 return
            return dp[i][j][index];
        if(arr[i][j] != target.charAt(index)) // 찾는글자가 아니다. 0으로
            return dp[i][j][index]=0;
 
        dp[i][j][index]=0;
 
        // 4자리 글자 체크
        for (int d = 0; d <4 ; d++) {
            for (int k = 1; k <= K ; k++) {
                int newX = j+moveX[d]*k;
                int newY = i+moveY[d]*k;
 
                // 맵의 범위를 벗어나거나 다음 문자가 정답 문자열과 같을 경우 다음 DFS 진행
                if(0<=newY && newY<&& 0<=newX && newX<&& arr[newY][newX] == target.charAt(index+1)){
                    dp[i][j][index] += dfs(newY, newX, index+1);
                }
 
            }
        }
        return dp[i][j][index];
    }
}
 
 

 

 

결과