力扣每日一题208:实现 Trie (前缀树)

2024-04-30 04:20

本文主要是介绍力扣每日一题208:实现 Trie (前缀树),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目内容

难度 中等

Trie(发音类似 "try")或者说 前缀树 是一种树形数据结构,用于高效地存储和检索字符串数据集中的键。这一数据结构有相当多的应用情景,例如自动补完和拼写检查。

请你实现 Trie 类:

  • Trie() 初始化前缀树对象。
  • void insert(String word) 向前缀树中插入字符串 word 。
  • boolean search(String word) 如果字符串 word 在前缀树中,返回 true(即,在检索之前已经插入);否则,返回 false 。
  • boolean startsWith(String prefix) 如果之前已经插入的字符串 word 的前缀之一为 prefix ,返回 true ;否则,返回 false 。

示例:

输入
["Trie", "insert", "search", "search", "startsWith", "insert", "search"]
[[], ["apple"], ["apple"], ["app"], ["app"], ["app"], ["app"]]
输出
[null, null, true, false, true, null, true]解释
Trie trie = new Trie();
trie.insert("apple");
trie.search("apple");   // 返回 True
trie.search("app");     // 返回 False
trie.startsWith("app"); // 返回 True
trie.insert("app");
trie.search("app");     // 返回 True

提示:

  • 1 <= word.length, prefix.length <= 2000
  • word 和 prefix 仅由小写英文字母组成
  • insertsearch 和 startsWith 调用次数 总计 不超过 3 * 104 次

面试中遇到过这道题?

1/5

通过次数

326.7K

提交次数

454K

通过率

72.0%

题目提供的结构(c++)

class Trie {
public:Trie() {}void insert(string word) {}bool search(string word) {}bool startsWith(string prefix) {}
};/*** 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);*/

思路

首先很多人还不知道前缀树。

简单说来就是,根节点不存放数据,其它节点每个节点存放一个字母。从根节点到某个节点形成的路径就是一个单词。每一个非根节点会有个标记,记录以该节点为尾的路径是不是一个单词。

首先每个节点的数据结构一定要有结束标志和表示孩子节点的数据(我用的是哈希表,键对应的是孩子节点的字符,值对应的是孩子节点的地址)。
其次还可以添加代表该节点的字母(我用哈希表表示孩子节点,所以可以通过父节点来找到当前节点的字母。如果不能通过父节点来表示当前节点的字母,那么一定要有个变量能专门表示当前节点的字母),根节点走到当前节点的值,根节点到当前节点的距离等等。
随后就是根据前缀树的特点来模拟。其中我认为最核心的算法就是以下部分。

int n=word.length();
Node* cur=root;
int i=0;
while( i<n&&(cur->child).find(word[i])!=(cur->child).end() ){cur=(cur->child)[word[i]];i++;
}

上述代码走完后 :
i会指向word中最后一个匹配成功的后一个位置
cur会指向前缀树中最后一个匹配成功的地址。
随后就很方便的进行插入,查找,找到前缀操作。

复杂度
时间复杂度:

添加和查找时间复杂度,: O(m)O(m)O(m)
其中m为word的长度
查找前缀的时间复杂度: O(p)O(p)O(p)
p为前缀的长度

空间复杂度:

空间复杂度为单词的总长度: O(numberofwords∗averagewordlength)

实现代码

class Trie {
public:struct Node{bool flag=false;        //结束标志,true代表是结束char letter;         //该节点代表的字母unordered_map<char,Node*> child;string val;         //根走到当前节点的值Node() {};Node(char c) : letter(c) {};};Node* root;Trie() {root=new Node('#');}void insert(string word) {int n=word.length();Node* cur=root;int i=0;while( i<n&&(cur->child).find(word[i])!=(cur->child).end() ){cur=(cur->child)[word[i]];i++;}//word已经存在的情况if(i==n&&(cur->flag)==true){return;}else if(i==n&&(cur->flag)==false){//word在前缀树中但是不存在的情况cur->flag=true;}else if(i!=n){while(i<n){(cur->child)[word[i]]=new Node(word[i]);cur=(cur->child)[word[i]];i++;}(cur->flag)=true;//标志结束}//}bool search(string word) {int n=word.length();int i=0;            //word中最后一个匹配成功的后一个位置Node* cur=root;     //前缀树中匹配成功的最后一个位置while( i<n&&(cur->child).find(word[i])!=(cur->child).end() ){cur=(cur->child)[word[i]];i++;}if(i==n&&(cur->flag)==true ) return true;else return false;}bool startsWith(string prefix) {int n=prefix.length();int i=0;Node* cur=root;while( i<n&&(cur->child).find(prefix[i])!=(cur->child).end() ){cur=(cur->child)[prefix[i]];i++;}if( i==n&&(cur->flag==true||(cur->child).size()!=0) ) return true;else return false;}
};/*** 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);*/

这篇关于力扣每日一题208:实现 Trie (前缀树)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

使用animation.css库快速实现CSS3旋转动画效果

《使用animation.css库快速实现CSS3旋转动画效果》随着Web技术的不断发展,动画效果已经成为了网页设计中不可或缺的一部分,本文将深入探讨animation.css的工作原理,如何使用以及... 目录1. css3动画技术简介2. animation.css库介绍2.1 animation.cs

Java进行日期解析与格式化的实现代码

《Java进行日期解析与格式化的实现代码》使用Java搭配ApacheCommonsLang3和Natty库,可以实现灵活高效的日期解析与格式化,本文将通过相关示例为大家讲讲具体的实践操作,需要的可以... 目录一、背景二、依赖介绍1. Apache Commons Lang32. Natty三、核心实现代

SpringBoot实现接口数据加解密的三种实战方案

《SpringBoot实现接口数据加解密的三种实战方案》在金融支付、用户隐私信息传输等场景中,接口数据若以明文传输,极易被中间人攻击窃取,SpringBoot提供了多种优雅的加解密实现方案,本文将从原... 目录一、为什么需要接口数据加解密?二、核心加解密算法选择1. 对称加密(AES)2. 非对称加密(R

基于Go语言实现Base62编码的三种方式以及对比分析

《基于Go语言实现Base62编码的三种方式以及对比分析》Base62编码是一种在字符编码中使用62个字符的编码方式,在计算机科学中,,Go语言是一种静态类型、编译型语言,它由Google开发并开源,... 目录一、标准库现状与解决方案1. 标准库对比表2. 解决方案完整实现代码(含边界处理)二、关键实现细

python通过curl实现访问deepseek的API

《python通过curl实现访问deepseek的API》这篇文章主要为大家详细介绍了python如何通过curl实现访问deepseek的API,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编... API申请和充值下面是deepeek的API网站https://platform.deepsee

SpringBoot实现二维码生成的详细步骤与完整代码

《SpringBoot实现二维码生成的详细步骤与完整代码》如今,二维码的应用场景非常广泛,从支付到信息分享,二维码都扮演着重要角色,SpringBoot是一个非常流行的Java基于Spring框架的微... 目录一、环境搭建二、创建 Spring Boot 项目三、引入二维码生成依赖四、编写二维码生成代码五

Java如何根据文件名前缀自动分组图片文件

《Java如何根据文件名前缀自动分组图片文件》一大堆文件(比如图片)堆在一个目录下,它们的命名规则遵循一定的格式,混在一起很难管理,所以本文小编就和大家介绍一下如何使用Java根据文件名前缀自动分组图... 目录需求背景分析思路实现代码输出结果知识扩展需求一大堆文件(比如图片)堆在一个目录下,它们的命名规

MyBatisX逆向工程的实现示例

《MyBatisX逆向工程的实现示例》本文主要介绍了MyBatisX逆向工程的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录逆向工程准备好数据库、表安装MyBATisX插件项目连接数据库引入依赖pom.XML生成实体类、

C#实现查找并删除PDF中的空白页面

《C#实现查找并删除PDF中的空白页面》PDF文件中的空白页并不少见,因为它们有可能是作者有意留下的,也有可能是在处理文档时不小心添加的,下面我们来看看如何使用Spire.PDFfor.NET通过C#... 目录安装 Spire.PDF for .NETC# 查找并删除 PDF 文档中的空白页C# 添加与删

Java实现MinIO文件上传的加解密操作

《Java实现MinIO文件上传的加解密操作》在云存储场景中,数据安全是核心需求之一,MinIO作为高性能对象存储服务,支持通过客户端加密(CSE)在数据上传前完成加密,下面我们来看看如何通过Java... 目录一、背景与需求二、技术选型与原理1. 加密方案对比2. 核心算法选择三、完整代码实现1. 加密上