반응형
이문제는 문자열 패턴찾기 문제이다.
단순히 일반적으로 찾으면 시간초과가 생겨서 풀수 없는 문제이다. 그렇기에 대표적인 문자열 알고리즘의 대표적인 KMP를 사용하였다. KMP의 알고리즘의 간단한 원리는 일단 패턴문자열에 대해 해당 문자열의 위치마다에서 접두사와 접미사가 최대로 같아지는 것을 카운팅해주는 배열을 만든다음, 그 배열을 이용하여 접두사와 접미사를 확인해가며 일정간격을 건너뛰어 시간복잡도가 거의 O(n)에 가깝도록 해결하는 방식이다.
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
|
package study5;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.List;
public class Main_bj_1786_찾기 {
static int count;
static List<Integer> list;
public static void main(String[] args) throws Exception {
BufferedReader br= new BufferedReader(new InputStreamReader(System.in));
String str=br.readLine();
String pattern=br.readLine();
count=0;
list=new ArrayList<>();
kmp(str,pattern);
System.out.println(count);
for(int num : list) {
System.out.println(num);
}
}
static void kmp(String str, String pattern) {
int[] p = find(pattern);
// System.out.println(Arrays.toString(p));
int j=0;
for(int i=0;i<str.length();i++) {
while(j>0 && str.charAt(i)!=pattern.charAt(j)) {
j=p[j-1];
}
if(str.charAt(i)==pattern.charAt(j)) {
if(j==pattern.length()-1) {
count++;
list.add(i-pattern.length()+2);
// System.out.println(i-pattern.length()+2);
j=p[j];
}else {
j++;
}
}
}
}
static int[] find(String pattern) {
int[] p =new int[pattern.length()];
int j=0;
for(int i=1;i<pattern.length();i++) {
while(j>0 && pattern.charAt(i) != pattern.charAt(j)) {
j=p[j-1];
}
if(pattern.charAt(i)==pattern.charAt(j)) {
j++;
p[i]=j;
}
}
return p;
}
}
|
반응형
'Algorithm' 카테고리의 다른 글
[BOJ] 1541. 잃어버린 괄호 (0) | 2020.09.05 |
---|---|
[Programmers] 방금그곡 (0) | 2020.09.04 |
[Programmers] 괄호 변환 (0) | 2020.09.03 |
[BOJ] 2293. 동전 1 (0) | 2020.09.02 |
[Programmers] 경주로 건설 (0) | 2020.08.31 |