Algorithm/Baekjoon
[백준 9202 : JAVA] Boggle / 트라이 + DFS
팡트루야
2021. 3. 14. 00:50
문제
풀이
푸는데 반나절걸렸다.. (한 5시간즘..?) 그래도 혼자 힘으로 풀어서 뿌듯하다 ..
일단 단어 사전에 들어가는 단어들은 트라이 구조에 담았다. 일반적인 트라이 방법대로 구현하였고, 한 가지 추가한 부분이 있다면 기존의 find() 메서드에서 bool 타입으로 완전한 문자열을 검색할건지에 대한 파라미터를 받았다.
boolean find(String key, boolean isCompleteString) {
// ...
}
원래의 find() 메서드의 역할은 해당 문자열이 있는지를 검색하는 것인데, 이 기능을 확장해 접두사 문자열에 대해서도 확인할 수 있게 했다.
{"Hello", "He"} 두 개의 문자열이 있을 때, 완전한 문자열 검색이라면 "Hel" 이라는 문자열은 존재하지 않는 문자열이라 실패한다. 하지만, 접두사로 확장하면 "Hello" 라는 문자열의 접두사 중 하나가 "Hel" 이기 때문에 성공한다.
다음으로, Boggle 맵을 입력받은 후 DFS로 완전 탐색을 진행하였다.
코드
import java.io.*;
import java.util.*;
public class Main {
private static char[][] boggleMap;
private static boolean[][] visited;
private static int[] dx = {0, 0, -1, 1, 1, 1, -1, -1};
private static int[] dy = {-1, 1, 0, 0, 1, -1, 1, -1};
private static Trie trie = new Trie();
private static StringBuilder sb = new StringBuilder();
public static void main(String[] args) {
try (BufferedReader br = new BufferedReader(new InputStreamReader(System.in))) {
int w = Integer.parseInt(br.readLine());
while (w-- > 0) {
trie.insert(br.readLine());
}
br.readLine();
int b = Integer.parseInt(br.readLine());
while (b-- > 0) {
boggleMap = new char[4][4];
visited = new boolean[4][4];
for (int i = 0; i < 4; i++) {
String line = br.readLine();
for (int j = 0; j < 4; j++) {
boggleMap[i][j] = line.charAt(j);
}
}
Set<String> stringSet = new HashSet<>();
for (int y = 0; y < 4; y++) {
for (int x = 0; x < 4; x++) {
visited[y][x] = true;
dfs(x, y, boggleMap[y][x] + "", stringSet);
visited[y][x] = false;
}
}
setPrint(stringSet);
if (b != 0) br.readLine();
}
System.out.println(sb.toString());
} catch (Exception ex) {
ex.getMessage();
}
}
private static void setPrint(Set<String> stringSet) {
List<String> list = new ArrayList<>(stringSet);
Collections.sort(list);
String longestStr = "";
int score = 0;
for (String s : list) {
if (s.length() > longestStr.length()) {
longestStr = s;
}
score += getScore(s.length());
}
sb.append(score).append(" ").append(longestStr).append(" ").append(list.size()).append("\n");
}
private static int getScore(int maxLen) {
int score = 0;
if (maxLen <= 2) score = 0;
else if (maxLen <= 4) score = 1;
else if (maxLen == 5) score = 2;
else if (maxLen == 6) score = 3;
else if (maxLen == 7) score = 5;
else if (maxLen == 8) score = 11;
return score;
}
private static void dfs(int x, int y, String str, Set<String> stringSet) {
if (trie.find(str, true)) { // 만약 문자열의 끝으로 설정되어있다면, set에 단어를 추가한다.
stringSet.add(str);
}
for (int i = 0; i < 8; i++) {
int nx = x + dx[i];
int ny = y + dy[i];
if (nx < 0 || nx >= 4 || ny < 0 || ny >= 4) continue;
if (visited[ny][nx]) continue;
String key = str + boggleMap[ny][nx]; // 있는지 확인해야할 문자열
if (trie.find(key, false)) { // 해당 좌표까지의 문자열이 있다면, 다음 문자 탐색
visited[ny][nx] = true;
dfs(nx, ny, key, stringSet);
visited[ny][nx] = false;
}
}
}
private static class Trie {
TrieNode root = new TrieNode();
private void insert(String key) {
int length = key.length();
TrieNode currentNode = root;
for (int i = 0; i < length; i++) {
int next = key.charAt(i) - 'A';
if (currentNode.child[next] == null) {
currentNode.child[next] = new TrieNode();
currentNode.nChild++;
}
currentNode = currentNode.child[next];
}
currentNode.isFinish = true;
}
// isCompleteString 값에 따른 의미
// true: 완전한 문자열을 검색할 경우
// false: 해당 문자열까지의 노드가 존재하는지만 알고 싶은 경우 (즉, 접두사를 포함)
private boolean find(String key, boolean isCompleteString) {
int length = key.length();
TrieNode currentNode = root;
for (int i = 0; i < length; i++) {
int next = key.charAt(i) - 'A';
if (currentNode.child[next] == null) return false;
currentNode = currentNode.child[next];
}
return isCompleteString ? currentNode.isFinish : true;
}
}
private static class TrieNode {
TrieNode[] child = new TrieNode[26]; // 알파벳 대문자로만 이루어져 있다.
boolean isFinish = false;
int nChild = 0; // 자식 노드의 갯수
}
}