Welcome to Subscribe On Youtube
Question
Formatted question description: https://leetcode.ca/all/208.html
A trie (pronounced as "try") or prefix tree is a tree data structure used to efficiently store and retrieve keys in a dataset of strings. There are various applications of this data structure, such as autocomplete and spellchecker.
Implement the Trie class:
Trie()
Initializes the trie object.void insert(String word)
Inserts the stringword
into the trie.boolean search(String word)
Returnstrue
if the stringword
is in the trie (i.e., was inserted before), andfalse
otherwise.boolean startsWith(String prefix)
Returnstrue
if there is a previously inserted stringword
that has the prefixprefix
, andfalse
otherwise.
Example 1:
Input ["Trie", "insert", "search", "search", "startsWith", "insert", "search"] [[], ["apple"], ["apple"], ["app"], ["app"], ["app"], ["app"]] Output [null, null, true, false, true, null, true] Explanation Trie trie = new Trie(); trie.insert("apple"); trie.search("apple"); // return True trie.search("app"); // return False trie.startsWith("app"); // return True trie.insert("app"); trie.search("app"); // return True
Constraints:
1 <= word.length, prefix.length <= 2000
word
andprefix
consist only of lowercase English letters.- At most
3 * 104
calls in total will be made toinsert
,search
, andstartsWith
.
Algorithm
Each node of the alphabetic dictionary tree defines an array of child node pointers with a size of 26
, and then uses a marker to record whether it is a word up to the current position.
When initializing, the 26
child nodes are all empty.
Then the insert()
operation only needs to calculate the position of each character of the string to be inserted, and then find out whether this child node exists, if it does not exist, create a new one, and then move to the next one.
The word search()
and prefix search operations are very similar to the insert operation. The difference is that if there are no child nodes, false is returned. At the end of the search time, you need to look at the flag, and you can directly return true when you search the prefix.
Code
-
public class Implement_Trie_Prefix_Tree { public static void main(String[] args) { Implement_Trie_Prefix_Tree out = new Implement_Trie_Prefix_Tree(); Trie trie = out.new Trie(); trie.insert("abc"); trie.insert("ab"); System.out.println(trie.search("ab")); trie.insert("ab"); System.out.println(trie.search("ab")); System.out.println(trie.startsWith("a")); } class TrieNode { // R children to node children private TrieNode[] children; private final int R = 26; private boolean isEnd; public TrieNode() { children = new TrieNode[R]; } public boolean containsKey(char ch) { return children[ch - 'a'] != null; } public TrieNode get(char ch) { return children[ch - 'a']; } public void put(char ch, TrieNode node) { children[ch - 'a'] = node; } public void setEnd() { isEnd = true; } public boolean isEnd() { return isEnd; } } class Trie { private TrieNode root; public Trie() { root = new TrieNode(); } // Inserts a word into the trie. public void insert(String word) { TrieNode node = root; // @note:@memorize: iteration is better than recursion during trie building for (int i = 0; i < word.length(); i++) { char currentChar = word.charAt(i); if (!node.containsKey(currentChar)) { node.put(currentChar, new TrieNode()); } node = node.get(currentChar); } node.setEnd(); } // search a prefix or whole key in trie and // returns the node where search ends private TrieNode searchPrefix(String word) { TrieNode node = root; for (int i = 0; i < word.length(); i++) { char curLetter = word.charAt(i); if (node.containsKey(curLetter)) { node = node.get(curLetter); } else { return null; } } return node; } // Returns if the word is in the trie. public boolean search(String word) { TrieNode node = searchPrefix(word); return node != null && node.isEnd(); } // Returns if there is any word in the trie // that starts with the given prefix. public boolean startsWith(String prefix) { TrieNode node = searchPrefix(prefix); return node != null; } } // Your Trie object will be instantiated and called as such: // Trie trie = new Trie(); // trie.insert("somestring"); // trie.search("key"); } ############ class Trie { private Trie[] children; private boolean isEnd; public Trie() { children = new Trie[26]; } public void insert(String word) { Trie node = this; for (char c : word.toCharArray()) { int idx = c - 'a'; if (node.children[idx] == null) { node.children[idx] = new Trie(); } node = node.children[idx]; } node.isEnd = true; } public boolean search(String word) { Trie node = searchPrefix(word); return node != null && node.isEnd; } public boolean startsWith(String prefix) { Trie node = searchPrefix(prefix); return node != null; } private Trie searchPrefix(String s) { Trie node = this; for (char c : s.toCharArray()) { int idx = c - 'a'; if (node.children[idx] == null) { return null; } node = node.children[idx]; } return node; } } /** * Your Trie object will be instantiated and called as such: * Trie obj = new Trie(); * obj.insert(word); * boolean param_2 = obj.search(word); * boolean param_3 = obj.startsWith(prefix); */
-
// OJ: https://leetcode.com/problems/implement-trie-prefix-tree/ // Time: O(W) for insert/search/startsWith // Space: O(1) extra space for all struct TrieNode { TrieNode *next[26] = {}; bool word = false; }; class Trie { TrieNode root; TrieNode *find(string &word) { auto node = &root; for (char c : word) { if (!node->next[c - 'a']) return NULL; node = node->next[c - 'a']; } return node; } public: void insert(string word) { auto node = &root; for (char c : word) { if (!node->next[c - 'a']) node->next[c - 'a'] = new TrieNode(); node = node->next[c - 'a']; } node->word = true; } bool search(string word) { auto node = find(word); return node && node->word; } bool startsWith(string prefix) { return find(prefix); } };
-
import collections class TrieNode: def __init__(self): ''' Usually, a Python dictionary throws a KeyError if you try to get an item with a key that is not currently in the dictionary. The defaultdict in contrast will simply create any items that you try to access https://stackoverflow.com/questions/5900578/how-does-collections-defaultdict-work ''' self.child = collections.defaultdict(TrieNode) self.is_word = False class Trie: def __init__(self): self.root = TrieNode() def insert(self, word: str) -> None: cur = self.root for letter in word: cur = cur.child[letter] cur.is_word = True def search(self, word: str) -> bool: cur = self.root for letter in word: # cur = cur.child[letter] ==> will not work, it will creat a default node for this letter cur = cur.child.get(letter) if not cur: return False return cur.is_word def startsWith(self, prefix: str) -> bool: cur = self.root for letter in prefix: # cur = cur.child[letter] ==> will not work, it will creat a default node for this letter cur = cur.child.get(letter) if not cur: return False return True # Your Trie object will be instantiated and called as such: # obj = Trie() # obj.insert(word) # param_2 = obj.search(word) # param_3 = obj.startsWith(prefix) ############ class Trie: def __init__(self): self.children = [None] * 26 self.is_end = False def insert(self, word: str) -> None: node = self for c in word: idx = ord(c) - ord('a') if node.children[idx] is None: node.children[idx] = Trie() node = node.children[idx] node.is_end = True def search(self, word: str) -> bool: node = self._search_prefix(word) return node is not None and node.is_end def startsWith(self, prefix: str) -> bool: node = self._search_prefix(prefix) return node is not None def _search_prefix(self, prefix: str): node = self for c in prefix: idx = ord(c) - ord('a') if node.children[idx] is None: return None node = node.children[idx] return node # Your Trie object will be instantiated and called as such: # obj = Trie() # obj.insert(word) # param_2 = obj.search(word) # param_3 = obj.startsWith(prefix) # below is using reduce() class Trie(object): def __init__(self): T = lambda: collections.defaultdict(T) self.root = T() def insert(self, word): reduce(dict.__getitem__, word, self.root)['#'] = True def search(self, word): return '#' in reduce(lambda cur, c: cur.get(c, {}), word, self.root) def startsWith(self, prefix): return bool(reduce(lambda cur, c: cur.get(c, {}), prefix, self.root))
-
type Trie struct { children [26]*Trie isEnd bool } func Constructor() Trie { return Trie{} } func (this *Trie) Insert(word string) { node := this for _, c := range word { idx := c - 'a' if node.children[idx] == nil { node.children[idx] = &Trie{} } node = node.children[idx] } node.isEnd = true } func (this *Trie) Search(word string) bool { node := this.SearchPrefix(word) return node != nil && node.isEnd } func (this *Trie) StartsWith(prefix string) bool { node := this.SearchPrefix(prefix) return node != nil } func (this *Trie) SearchPrefix(s string) *Trie { node := this for _, c := range s { idx := c - 'a' if node.children[idx] == nil { return nil } node = node.children[idx] } return node } /** * Your Trie object will be instantiated and called as such: * obj := Constructor(); * obj.Insert(word); * param_2 := obj.Search(word); * param_3 := obj.StartsWith(prefix); */
-
class TrieNode { children; isEnd; constructor() { this.children = new Array(26); this.isEnd = false; } } class Trie { root; constructor() { this.root = new TrieNode(); } insert(word: string): void { let head = this.root; for (let char of word) { let index = char.charCodeAt(0) - 97; if (!head.children[index]) { head.children[index] = new TrieNode(); } head = head.children[index]; } head.isEnd = true; } search(word: string): boolean { let head = this.searchPrefix(word); return head != null && head.isEnd; } startsWith(prefix: string): boolean { return this.searchPrefix(prefix) != null; } private searchPrefix(prefix: string) { let head = this.root; for (let char of prefix) { let index = char.charCodeAt(0) - 97; if (!head.children[index]) return null; head = head.children[index]; } return head; } }
-
/** * Initialize your data structure here. */ var Trie = function () { this.children = {}; }; /** * Inserts a word into the trie. * @param {string} word * @return {void} */ Trie.prototype.insert = function (word) { let node = this.children; for (let char of word) { if (!node[char]) { node[char] = {}; } node = node[char]; } node.isEnd = true; }; /** * Returns if the word is in the trie. * @param {string} word * @return {boolean} */ Trie.prototype.search = function (word) { let node = this.searchPrefix(word); return node != undefined && node.isEnd != undefined; }; Trie.prototype.searchPrefix = function (prefix) { let node = this.children; for (let char of prefix) { if (!node[char]) return false; node = node[char]; } return node; }; /** * Returns if there is any word in the trie that starts with the given prefix. * @param {string} prefix * @return {boolean} */ Trie.prototype.startsWith = function (prefix) { return this.searchPrefix(prefix); }; /** * Your Trie object will be instantiated and called as such: * var obj = new Trie() * obj.insert(word) * var param_2 = obj.search(word) * var param_3 = obj.startsWith(prefix) */
-
public class Trie { bool isEnd; Trie[] children = new Trie[26]; public Trie() { } public void Insert(string word) { Trie node = this; foreach (var c in word) { var idx = c - 'a'; node.children[idx] ??= new Trie(); node = node.children[idx]; } node.isEnd = true; } public bool Search(string word) { Trie node = SearchPrefix(word); return node != null && node.isEnd; } public bool StartsWith(string prefix) { Trie node = SearchPrefix(prefix); return node != null; } private Trie SearchPrefix(string s) { Trie node = this; foreach (var c in s) { var idx = c - 'a'; if (node.children[idx] == null) { return null; } node = node.children[idx]; } return node; } } /** * Your Trie object will be instantiated and called as such: * Trie obj = new Trie(); * obj.Insert(word); * bool param_2 = obj.Search(word); * bool param_3 = obj.StartsWith(prefix); */