본문 바로가기
Algorithm/문제 풀이

[알고리즘] 전화번호 목록 (백준 5052번)

by Sky Titan 2020. 8. 26.
728x90
 

5052번: 전화번호 목록

문제 전화번호 목록이 주어진다. 이때, 이 목록이 일관성이 있는지 없는지를 구하는 프로그램을 작성하시오. 전화번호 목록이 일관성을 유지하려면, 한 번호가 다른 번호의 접두어인 경우가 없�

www.acmicpc.net

 문자열 처리 문제인데 전화번호들의 일관성을 유지하려면 '한 번호가 다른 번호의 접두어인 경우가 없어야 한다' 라고 되어있다. 즉 911, 9112584 같은 경우엔 일관성이 없는 경우이다.

 

 접두어 판별문제인만큼 Trie를 사용해야한다. Trie 자료구조에 관한 것은 알고리즘 설명 카테고리에 올라와 있을 것이다.

Trie 클래스와 Node 클래스를 만들고 Trie 클래스에 String을 삽입하는 insertString() 메서드와 일관성 있는지 판별하는 isConsistent() 메서드를 만들면 끝이다.

 

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.util.*;

public class Main {

	static void solution() throws Exception
	{
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));

		int T = Integer.parseInt(br.readLine());

		for(int t = 0;t < T;t++)
		{
			int N = Integer.parseInt(br.readLine());

			StringBuilder list[] = new StringBuilder[N];
			Trie trie = new Trie();

			for(int i = 0;i < N;i++)
			{
				list[i] = new StringBuilder(br.readLine());
				trie.insertString(list[i]);
			}

			for(int i = 0;i < N;i++)
			{
				if(!trie.isConsistent(list[i]))
				{
					bw.write("NO\n");
					break;
				}

				if(i == N-1)
					bw.write("YES\n");
			}
		}
		bw.close();
	}

	static class Trie{

		Node root = new Node();

		public Trie()
		{

		}

		public void insertString(StringBuilder str)
		{
			Node node = root;

			for(int i = 0;i < str.length();i++)
			{
				char c = str.charAt(i);

				node.children.putIfAbsent(c, new Node());
				node = node.children.get(c);

				if(i == str.length()-1)
					node.isFinish = true;
			}
		}

		public boolean isConsistent(StringBuilder str)
		{
			Node node = root;

			for(int i = 0;i < str.length();i++)
			{
				char c = str.charAt(i);

				node = node.children.get(c);

				//아직 끝에 도달하지 않았는데 문자열이 존재한다면(접두어) 일관성 파괴
				if(i < str.length()-1)
				{
					if(node.isFinish)
						return false;
				}
			}

			return true;
		}
	}

	static class Node{

		HashMap<Character, Node> children = new HashMap<>();
		boolean isFinish = false;//현재까지의 문자열이 존재하는 지 판별

		public Node()
		{

		}
	}


	public static void main(String[] args) {

		try
		{
			solution();
		}
		catch (Exception e)
		{
			e.printStackTrace();
		}
	}


}
728x90

댓글