算法第十八天-实现Trie(前缀树)

2024-01-14 10:44

本文主要是介绍算法第十八天-实现Trie(前缀树),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

实现Trie(前缀树)

题目要求

解题思路

本文是前缀入门教程
从二叉树说起
前缀树,也是一种树。为了理解前缀树,我们先从二叉树说起。常见的二叉树结构是下面这样子的:

class TreeNode { int val; TreeNode* left; TreeNode* right; }

可以看到一个树的节点包含了三个元素:该节点本身的值,左子树的指针,右子树的指针。二叉树可视化是下面这样子的:

二叉树的每个节点只有两个孩子,那如果每个节点可以有多个孩子呢?这就形成了多叉树。多叉树的子节点数目一般不是固定的,所以会用变长数组来保存所有的子节点的指针。多叉树的结构式下面这样:

class TreeNode { int val; vector<TreeNode*> children; 
}

多叉树可视化是下面这样:

对于普通的多叉树,每个节点的所有子节点可能是没有任何规律的。而本题讨论的[前缀树]就是每个节点的Children有规律的多叉树。

前缀树
(只保存小写字符的)[前缀树]是一种特殊的多叉树,它的TrieNode中Children是一个大小为26的一维数组,分别对应了26个英文字母,也就是说形成了一棵26叉树。
前缀树的结果可以定义为下面这样。
里面存储了两个信息:

  • isWord表示从根节点到当前节点为止,该路径是否已经形成了一个有效的字符串。
  • children是该节点的所有子节点
class TrieNode {
public:vector<TrieNode*> children;bool isWord;TrieNode() : isWord(false), children(26, nullptr) {}~TrieNode() {for (auto& c : children)delete c;}
};

构建
在构建前缀树的时候,按照下面的方法:

  • 根节点不保存任何信息;
  • 关键词放到[前缀树]时,需要把它拆成各个字符,每个字符按照其在'a'~'z'的序号,放在对应的children里面,下一个字符实在当前字符的子节点。
  • 一个输入字符串构建[前缀树]结束的时候,需要把该节点的isword标记为true,说明从根节点到当前节点的路径,构成了一个关键词。

看下面这个图的时候,需要注意:
1.所有以相同字符开头的字符串,会聚合到同一个子树上。比如{'am','an','as'}
2.并不一定是到达叶子节点才形成一个关键词,只要isword为true,那么从根节点到当前节点的路劲就是关键词。比如{'c','cv'}
在这里插入图片描述

有些题解把字符画在节点中,这是不准确的。因为前缀树是根据字符在children中的位置确定子树,而不真正在书中存储了'a'~'z'这些字符。树中每个节点存储的isWord,表示从根节点到当前节点的路径是否构成了一个关键词。

查询
在判断一个关键词是否在[前缀树]中时,需要依次遍历该关键词所有字符,在前缀树中找到这条路径。可能会出现三种情况:
1.在寻找路径的过程中,发现到某个位置路径断了。比如在上面的前缀树图中寻找'd'或者''ar或者'any',由于树中没有构建对应的节点,那么就查找不到这些关键词;
2.找到了这条路径,但是最后一个节点的isWord为false。这也说明没有改关键词。比如在上面的前缀树图中寻找'a';
3.找到了这条路径,并且最后一个节点的isWord为true。这说明前缀树存储了这个关键词,比如上面前缀树图中的'am','cv'等。

应用
上面说了这么多前缀树,那前缀树有什么用那?
其实我们生活中就有应用。比如我们常见的电话拨号键盘,当我们输入一些数字的时候,后面会自动提示以我们的输入数字为开头的所有号码。

代码

下面的Python解法中,保存children是使用的字典,它保存的结构式{字符:Node},所以可以直接通过children[“a”]来获取当前节点的’a’子树。

class Node(object):def __init__(self):self.children = collections.defaultdict(Node)self.isword = False
class Trie:def __init__(self):"""Initialize your data structure here."""self.root = Node()def insert(self, word: str) -> None:"""Inserts a word into the trie."""current = self.rootfor w in word:current = current.children[w]current.isword = Truedef search(self, word: str) -> bool:"""Returns if the word is in the trie."""current = self.rootfor w in word:current = current.children.get(w)if current == None:return Falsereturn current.isworddef startsWith(self, prefix: str) -> bool:"""Returns if there is any word in the trie that starts with the given prefix."""current = self.rootfor w in prefix:current = current.children.get(w)if current == None:return Falsereturn 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)

复杂度分析

时间复杂度:初始化为 O ( 1 ) O(1) O(1),其余操作为 O ( ∣ S ∣ ) O(|S|) O(S),其中|S|是每次插入或咨询的字符串长度。
空间复杂度: O ( ∣ T ∣ ⋅ ∑ ) O(|T|·∑) O(T),其中|T|为所有插入字符串的长度之和,∑为字符集的大小,本题∑=26

这篇关于算法第十八天-实现Trie(前缀树)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/604884

相关文章

Java实现数据库图片上传与存储功能

《Java实现数据库图片上传与存储功能》在现代的Web开发中,上传图片并将其存储在数据库中是常见的需求之一,本文将介绍如何通过Java实现图片上传,存储到数据库的完整过程,希望对大家有所帮助... 目录1. 项目结构2. 数据库表设计3. 实现图片上传功能3.1 文件上传控制器3.2 图片上传服务4. 实现

Java实现MD5加密的四种方式

《Java实现MD5加密的四种方式》MD5是一种广泛使用的哈希算法,其输出结果是一个128位的二进制数,通常以32位十六进制数的形式表示,MD5的底层实现涉及多个复杂的步骤和算法,本文给大家介绍了Ja... 目录MD5介绍Java 中实现 MD5 加密方式方法一:使用 MessageDigest方法二:使用

mysql删除无用用户的方法实现

《mysql删除无用用户的方法实现》本文主要介绍了mysql删除无用用户的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 1、删除不用的账户(1) 查看当前已存在账户mysql> select user,host,pa

Nginx配置location+rewrite实现隐性域名配置

《Nginx配置location+rewrite实现隐性域名配置》本文主要介绍了Nginx配置location+rewrite实现隐性域名配置,包括基于根目录、条件和反向代理+rewrite配置的隐性... 目录1、配置基于根目录的隐性域名(就是nginx反向代理)2、配置基于条件的隐性域名2.1、基于条件

Linux配置IP地址的三种实现方式

《Linux配置IP地址的三种实现方式》:本文主要介绍Linux配置IP地址的三种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录环境RedHat9第一种安装 直接配置网卡文件第二种方式 nmcli(Networkmanager command-line

Java实现将Markdown转换为纯文本

《Java实现将Markdown转换为纯文本》这篇文章主要为大家详细介绍了两种在Java中实现Markdown转纯文本的主流方法,文中的示例代码讲解详细,大家可以根据需求选择适合的方案... 目录方法一:使用正则表达式(轻量级方案)方法二:使用 Flexmark-Java 库(专业方案)1. 添加依赖(Ma

使用EasyExcel实现简单的Excel表格解析操作

《使用EasyExcel实现简单的Excel表格解析操作》:本文主要介绍如何使用EasyExcel完成简单的表格解析操作,同时实现了大量数据情况下数据的分次批量入库,并记录每条数据入库的状态,感兴... 目录前言固定模板及表数据格式的解析实现Excel模板内容对应的实体类实现AnalysisEventLis

Mybatis从3.4.0版本到3.5.7版本的迭代方法实现

《Mybatis从3.4.0版本到3.5.7版本的迭代方法实现》本文主要介绍了Mybatis从3.4.0版本到3.5.7版本的迭代方法实现,包括主要的功能增强、不兼容的更改和修复的错误,具有一定的参考... 目录一、3.4.01、主要的功能增强2、selectCursor example3、不兼容的更改二、

如何使用C#串口通讯实现数据的发送和接收

《如何使用C#串口通讯实现数据的发送和接收》本文详细介绍了如何使用C#实现基于串口通讯的数据发送和接收,通过SerialPort类,我们可以轻松实现串口通讯,并结合事件机制实现数据的传递和处理,感兴趣... 目录1. 概述2. 关键技术点2.1 SerialPort类2.2 异步接收数据2.3 数据解析2.

mybatis-plus 实现查询表名动态修改的示例代码

《mybatis-plus实现查询表名动态修改的示例代码》通过MyBatis-Plus实现表名的动态替换,根据配置或入参选择不同的表,本文主要介绍了mybatis-plus实现查询表名动态修改的示... 目录实现数据库初始化依赖包配置读取类设置 myBATis-plus 插件测试通过 mybatis-plu