ProgramSoliving

프로그래머스 : 가사검색 JavsScript

하이후에호 2021. 7. 1. 01:09
반응형

자료구조 : 트라이

 

1년전 정리했던 자료구조이다. Java로 짜여져있는 내코드를 참조했다.

https://redbinalgorithm.tistory.com/167?category=880023 

 

트라이(Trie) : JAVA / 백준 5052

트라이 알고리즘 일반 적인 정수들은 O(1) 시간내에 비교가 가능합니다 . 1==1 ,123>4 처럼 단순연산으로 접근 할 수 있습니다. 하지만 String 같은경우에는 "ABC" == "ABCD" 를 고려하기 위해서는 최대 문

redbinalgorithm.tistory.com

 

트라이 알고리즘은 특정 자료안에 key가 존재하는가 여부를 O(NM) 안에 알 수 있는 가장 빠른 자료구조이다(N은 쿼리수)

 

이러한 특성때문에 해당 프로그래머스 문제에서도 쿼리수가 많은 경우에도 빠르게 문제를 찾을 수 있다.

 

문제의 아이디어 words를 정방향, 역방향으로 문자의 길이 배열별로 트라이 자료구조를 생성한다. 

- 길이가 5인 word 정방향, 역방향

- 길이가 3인 word 정방향, 역방향

 

이렇게 생성해도 최대 40000개의 배열이 생성된다.

 

트라이 자료구조에 현재 index(여기서는 word의 몇번째 단어) 분기에서 단어가 몇개있는지 저장을 한다.

insert 마다 count++를 해주면 해당 분기에서 몇개의 단어가 등록되어있는지 알 수 있다.

 

따라서 트라이 구조를 만들고 queries를 탐색할 때 정방향 , 역방향 알맞게 find를 돌린다면, 처음으로 ?가 나오는 순간에 현재 노드의 등록된 단어수를 반환한다. 만약 null값이 나온다면 사전에 등록되어 있지 않다는 뜻이니 0을 반환한다.

 

const ALPHABET = 26;

class TrieNode {
    constructor(){
        this.children = Array(ALPHABET).fill(null);
        this.isEndOfWord = false;
        this.count = 0;
    }
}

function insert(key, root) {
    let level;
    let length = key.length;
    let index;

    root.count++;
    let pCrawl = root;
    
    for(level=0; level<length; ++level) {
        index = key[level].charCodeAt(0) - 'a'.charCodeAt(0);
        if(pCrawl.children[index] === null)
            pCrawl.children[index] = new TrieNode();
        pCrawl.children[index].count++;
        pCrawl = pCrawl.children[index];
    }
    pCrawl.isEndOfWord = true;
}

function find(key, root) {
    let level;
    let length = key.length;
    let index;
    let pCrawl = root;
    for(level=0;level<length;++level) {
        if(key[level] === '?')
            return pCrawl.count;
        index = key[level].charCodeAt(0) - 'a'.charCodeAt(0);
        if(pCrawl.children[index] === null)
            return 0;

        pCrawl = pCrawl.children[index];
    }

    return pCrawl.count;
}

const frontRoot = Array.from({length:10001},() => new TrieNode());
const rearRoot = Array.from({length:10001},() => new TrieNode());

function solution(words, queries) {
    
    // 정방향, 반대방향 Trie 길이별 생성
    for(let word of words) {
        let len = word.length;
        insert(word,frontRoot[len]);
        insert(word.split("").reverse().join(""),rearRoot[len]);
    }
    var answer = [];
    // 정답 찾기 query
    for(let query of queries) {
        // 정방향
        let len = query.length;
        if(query[0] !== '?') {
            answer.push(find(query,frontRoot[len]));
        }
        // 역방향
        else {
            answer.push(find(query.split("").reverse().join(""),rearRoot[len]))
        }
    }
    return answer;
}
반응형