Competition/Baekjoon
[백준] 2186번 자바 문자판
bisi
2020. 4. 24. 13:20
문제 출처
https://www.acmicpc.net/problem/2186
접근 방식 및 풀이
- 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
|
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.StringTokenizer;
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<N && 0<=newX && newX<M && arr[newY][newX] == target.charAt(index+1)){
dp[i][j][index] += dfs(newY, newX, index+1);
}
}
}
return dp[i][j][index];
}
}
|
결과