【Hot 100】208. 实现 Trie (前缀树)
目录
- 引言
- 实现 Trie (前缀树)
- 我的解题
- 代码解析
- 代码思路分析
- 优化建议
- 1. 内存泄漏问题
- 2. 使用智能指针优化内存管理
- 3. 输入合法性校验(可选)
- 4. 其他优化
- 总结
- 🙋♂️ 作者:海码007
- 📜 专栏:算法专栏
- 💥 标题:【Hot 100】208. 实现 Trie (前缀树)
- ❣️ 寄语:书到用时方恨少,事非经过不知难!
引言
实现 Trie (前缀树)
- 🎈 题目链接:
- 🎈 做题状态:
我的解题
首先需要理解前缀树的定义,前缀树是一颗多叉树,树根不存储字母。每一层可能存储26个不同的字母。然后每一个单词对应这个多叉树的一条路径,并且路径的结尾会标识是单词的结尾。
class Trie {
private:bool isEnd;Trie* next[26]; //指针数组,有26个小写字母public:Trie() {isEnd = false;memset(next, 0, sizeof(next));}// 插入一个单词void insert(string word) {// node指向根节点并向下遍历Trie* node = this;for (char c : word){// 判断当前这个字母是否在当前层存在,如果不存在则创建一个新的树。if (node->next[c-'a'] == nullptr){node->next[c-'a'] = new Trie();}node = node->next[c-'a']; // 继续往下遍历}// 遍历到末尾后,需要标识 endnode->isEnd = true;}// 搜索当前单词是否存在,依次比较每个单词是否存在每一层中bool search(string word) {Trie* node = this;for (const char& c : word){if (node->next[c-'a'] == nullptr){return false;}node = node->next[c-'a'];}return node->isEnd; // 遍历到末尾后还需要判断是否是单词的结尾}// 判断这个前缀树是否包含 prefix 这个前缀bool startsWith(string prefix) {Trie* node = this;for (const char& c : prefix){if (node->next[c-'a'] == nullptr){return false;}node = node->next[c-'a'];}return true;}
};/*** 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);*/
代码解析
该代码实现了一个基本的前缀树(Trie)结构,支持插入、搜索和前缀匹配功能。以下是对代码的详细分析和优化建议:
代码思路分析
-
数据结构设计:
isEnd
:标记当前节点是否为单词的结尾。next[26]
:一个包含 26 个指针的数组,对应英文小写字母,用于构建字符到子节点的映射。
-
核心方法:
- 插入 (
insert
): 从根节点开始,逐字符向下遍历。若字符对应的子节点不存在,则创建新节点。遍历完成后,标记最后一个节点为单词结尾。 - 搜索 (
search
): 检查单词是否存在且最后一个节点被标记为结尾。 - 前缀匹配 (
startsWith
): 仅检查路径是否存在,无需判断结尾标记。
- 插入 (
-
正确性:
- 逻辑正确,符合前缀树的标准实现。
- 使用
memset
初始化指针数组,确保初始值为nullptr
。
优化建议
1. 内存泄漏问题
当前代码未释放动态分配的子节点,导致内存泄漏。需添加析构函数递归释放子节点:
~Trie() {for (int i = 0; i < 26; ++i) {delete next[i]; // 递归调用子节点的析构函数}
}
2. 使用智能指针优化内存管理
改用 std::unique_ptr
管理子节点,避免手动释放内存:
class Trie {
private:bool isEnd;std::array<std::unique_ptr<Trie>, 26> next; // 使用智能指针public:Trie() : isEnd(false) {} // 无需 memsetvoid insert(string word) {Trie* node = this;for (char c : word) {if (!node->next[c - 'a']) {node->next[c - 'a'] = std::make_unique<Trie>();}node = node->next[c - 'a'].get();}node->isEnd = true;}// search 和 startsWith 无需修改
};
3. 输入合法性校验(可选)
若需处理非法字符(如大写字母),可在插入/查询前添加校验:
void insert(string word) {for (char c : word) {if (c < 'a' || c > 'z') {throw std::invalid_argument("Invalid character");}}// 原插入逻辑
}
4. 其他优化
- 空间优化:使用压缩 Trie 树(如 Ternary Search Tree)减少内存占用,但会增加实现复杂度。
- 性能优化:当前时间复杂度为 O(L)(L 为字符串长度),已是最优,无需调整。
总结
代码逻辑正确,但存在内存泄漏问题。建议通过析构函数或智能指针优化内存管理。其他优化可根据实际需求选择。改进后的代码示例(使用智能指针)如下:
#include <memory> // 用于智能指针 unique_ptr
#include <array> // 用于固定大小的数组 array
#include <string> // 用于字符串操作class Trie {
private:// 标记当前节点是否为某个单词的结尾bool isEnd;// 使用智能指针管理子节点,避免内存泄漏// 数组大小为26,对应英文小写字母a-zstd::array<std::unique_ptr<Trie>, 26> next;public:// 构造函数:初始化 isEnd 为 false,表示初始时不是单词结尾// 智能指针数组 next 会自动初始化为 nullptrTrie() : isEnd(false) { }/*** 插入一个单词到 Trie 树中* @param word 待插入的单词*/void insert(const std::string& word) {// 从根节点(this)开始遍历Trie* node = this;// 逐个字符处理for (char c : word) {// 计算字符对应的索引(a->0, b->1, ..., z->25)int idx = c - 'a';// 如果当前字符的子节点不存在,则创建新节点if (node->next[idx] == nullptr) {node->next[idx] = std::make_unique<Trie>();}// 移动到子节点继续处理node = node->next[idx].get(); // get() 获取裸指针}// 标记单词的最后一个字符节点为结尾node->isEnd = true;}/*** 搜索 Trie 树中是否存在某个单词* @param word 待搜索的单词* @return 如果单词存在且完整匹配(最后一个字符是结尾),返回 true;否则返回 false*/bool search(const std::string& word) {// 从根节点开始遍历Trie* node = this;// 逐个字符检查for (const char& c : word) {int idx = c - 'a';// 如果当前字符的子节点不存在,说明单词不存在if (node->next[idx] == nullptr) {return false;}// 移动到子节点继续检查node = node->next[idx].get();}// 检查最后一个字符是否被标记为单词结尾return node->isEnd;}/*** 检查 Trie 树中是否存在某个前缀* @param prefix 待检查的前缀* @return 如果前缀存在(不要求是完整单词),返回 true;否则返回 false*/bool startsWith(const std::string& prefix) {// 从根节点开始遍历Trie* node = this;// 逐个字符检查for (const char& c : prefix) {int idx = c - 'a';// 如果当前字符的子节点不存在,说明前缀不存在if (node->next[idx] == nullptr) {return false;}// 移动到子节点继续检查node = node->next[idx].get();}// 只要路径存在,无论是否是单词结尾,都返回 truereturn true;}
};
相关文章:

【Hot 100】208. 实现 Trie (前缀树)
目录 引言实现 Trie (前缀树)我的解题代码解析代码思路分析优化建议1. 内存泄漏问题2. 使用智能指针优化内存管理3. 输入合法性校验(可选)4. 其他优化 总结 🙋♂️ 作者:海码007📜 专栏:算法专栏…...

【2025最新】Vm虚拟机中直接使用Ubuntu 免安装过程直接使用教程与下载
Ubuntu 是一个基于 Debian 的自由开源 Linux 操作系统,面向桌面、服务器和云计算平台广泛应用。 由英国公司 Canonical Ltd. 维护和发布,Ubuntu 强调易用性、安全性和稳定性,适合个人用户、开发者以及企业部署使用。 Ubuntu 默认使用 GNOME …...
思路解析:第一性原理解 SQL
目录 题目描述 🎯 应用第一性原理来思考这个 SQL 题目 ✅ 第一步:还原每个事件的本质单位 ✅ 第二步:如果一个表只有事件,如何构造事件对? ✅ 第三步:加过滤条件,只保留“同一机器、同一进…...
相机Camera日志分析之八:高通Camx HAL架构opencamera三级日志详解及关键字
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了:相机Camera日志分析之七:高通Camx HAL架构opencamera二级日志详解及关键字 这一篇我们开始讲: 相机Camera日志分析之八:高通Camx HAL架构opencamera三级日志详解及关键字 目录 【关注我,后续持续…...

Vue2 elementUI 二次封装命令式表单弹框组件
需求:封装一个表单弹框组件,弹框和表单是两个组件,表单组件以插槽的形式动态传入弹框组件中。 外部组件使用的方式如下: 直接上代码: MyDialog.vue 弹框组件 <template><el-dialog:titletitle:visible.syn…...
Docker入门教程:常用命令与基础概念
目录 简介常用命令Docker 常用命令汇总docker run 命令格式与参数解析 简介 Docker 是一个客户端-服务器(client-server)架构的应用程序,其中包含两个主要组件:Docker 客户端和 Docker 守护进程(也称为 Docker Daemon…...

Antd中Form详解:
1.获取Form表单值的方式: ① 使用Form.useForm()钩子(推荐方式) const [form] Form.useForm();const getFormValues () > {const values form.getFieldsValue();};<Form form{form}>...<Form.Item label{null}><Button onClick{ge…...
探索C语言中的二叉树:原理、实现与应用
一、引言 二叉树作为一种重要的数据结构,在计算机科学领域有着广泛的应用,无论是在操作系统的文件系统管理,还是在数据库的索引构建中,都能看到它的身影。在C语言中,我们可以利用指针灵活地构建和操作二叉树。接下来&…...

docker系列-DockerDesktop报错信息(Windows Hypervisor is not present)
Docker Desktop 报错信息 Docker Desktop - Windows Hypervisor is not present Docker Desktop is unable to detect a Hypervisor. Hardware assisted virtualization and data execution protection must be enabled in the BIOS.这是因为 Docker Desktop 需要启用 虚拟化技…...
03.Python 字符串中的空白字符处理
Python 字符串中的空白字符处理 什么是空白字符? 在处理字符串时,常常需要去除多余的空白字符。空白字符包括: 空格( )制表符(\t)换行符(\n)回车符(\r&#x…...

《基于 Kubernetes 的 WordPress 高可用部署实践:从 MariaDB 到 Nginx 反向代理》
手把手教你用 Kubernetes 部署高可用 WordPress 博客 本实验通过 Kubernetes 容器编排平台,完整部署了一个高可用的 WordPress 网站架构,包含 MariaDB 数据库、WordPress 应用和 Nginx 反向代理三大核心组件。实验涵盖了从基础环境准备到最终服务暴露的…...

Ubuntu源码版comfyui的安装
Comfyui也出桌面版了,但是想让大家多个人都使用怎么办呢?也有方法,安装Linux版,启动后会生成个网页地址,打开就能用了。 1、先来看下本地安装环境配置: 系统:Ubuntu 22.04 内存:2…...
多模态RAG与LlamaIndex——1.deepresearch调研
摘要 关键点: 多模态RAG技术通过结合文本、图像、表格和视频等多种数据类型,扩展了传统RAG(检索增强生成)的功能。LlamaIndex是一个开源框架,支持多模态RAG,提供处理文本和图像的模型、嵌入和索引功能。研…...
C++ 命令模式详解
命令模式(Command Pattern)是一种行为设计模式,它将请求封装为对象,从而使你可以参数化客户端使用不同的请求、队列或日志请求,以及支持可撤销的操作。 核心概念 设计原则 命令模式遵循以下设计原则: 单…...

制作一款打飞机游戏47:跳转
编辑器的问题 我们开始为不同的敌人编写一些行为,到目前为止进展顺利,一切都很棒。但上次我们遇到了一些问题,我们发现在这个编辑器中编写代码有时有点困难,因为当你想要在某行之间插入内容时,你不得不删除一切然后重…...

本地部署ollama及deepseek(linux版)
一、安装ollama export OLLAMA_MIRROR"https://ghproxy.cn/https://github.com/ollama/ollama/releases/latest/download"curl -fsSL https://ollama.com/install.sh | sed "s|https://ollama.com/download|$OLLAMA_MIRROR|g" | shexport OLLAMA_MIRROR&q…...
Java Spring Boot项目目录规范示例
以下是一个典型的 Java Spring Boot 项目目录结构规范示例,结合了分层架构和模块化设计的最佳实践: text 复制 下载 src/ ├── main/ │ ├── java/ │ │ └── com/ │ │ └── example/ │ │ └── myapp/ │…...
针对共享内存和上述windows消息机制 在C++ 和qt之间的案例 进行详细举例说明
针对共享内存和上述windows消息机制 在C++ 和qt之间的案例 进行详细举例说明 以下是关于在 C++ 和 Qt 中使用共享内存(QSharedMemory)和 Windows 消息机制(SendMessage / PostMessage)进行跨线程或跨进程通信的详细示例。 🧩 使用 QSharedMemory 进行进程间通信(Qt 示例…...

vue H5解决安卓手机软键盘弹出,页面高度被顶起
开发中安卓机上遇到的软键盘弹出导致布局问题 直接上代码_ 在这里插入代码片 <div class"container"><div class"appContainer" :style"{height:isKeyboardOpen? Heights :inherit}"><p class"name"><!-- 绑定…...

CSS专题之自定义属性
前言 石匠敲击石头的第 12 次 CSS 自定义属性是现代 CSS 的一个强大特性,可以说是前端开发需知、必会的知识点,本篇文章就来好好梳理一下,如果哪里写的有问题欢迎指出。 什么是 CSS 自定义属性 CSS 自定义属性英文全称是 CSS Custom Proper…...
问题 | 当前计算机视觉迫切解决的问题
当前计算机视觉领域虽然在技术上取得了显著进展,但仍面临一系列关键挑战。结合最新研究与应用现状,以下是最迫切需要解决的几大问题: 1. 数据质量与多样性不足 高质量标注数据的获取:训练高效模型依赖大量精准标注的数据&#x…...

七、深入 Hive DDL:管理表、分区与洞察元数据
作者:IvanCodes 日期:2025年5月13日 专栏:Hive教程 内容导航 一、表的 DDL 操作 (非创建)二、分区的 DDL 操作三、洞察元数据:SHOW 命令的威力结语:DDL 与 SHOW,Hive 管理的双翼练习题一、选择题二、代码题…...
Qt6.x检查网络是否在线(与Qt 5.x不同)
Qt 5.x.x 要判断客户端网络是否联通,一般用如下方法: #include <QNetworkConfigurationManager>auto netWorkCheck new QNetworkConfigurationManager(); auto flag netWorkCheck->isOnline(); Qt 6.x.x 废弃了 QNetworkConfigurationManag…...

直接在Excel中用Python Matplotlib/Seaborn/Plotly......
本次分享如何利用pyxll包,实现直接在Excel中使用Python Matplotlib/Seaborn/Plotly等强大可视化工具。 pyxll配置 pyxll安装 pip install pyxll pyxll install pyxll自定义方法 例如,自定义一个计算斐波那契数的方法fib,并使用pyxll装饰器…...

React面试常问问题详解
以下是30个React面试中常见的问题及简要解析,涵盖基础概念、核心原理、性能优化、Hooks、状态管理等方面,适用于初中高级开发者准备面试时参考: 一、React 基础与核心概念 React 是什么? React 是由 Facebook 开发的用于构建用户界…...

【Java】网络编程(Socket)
网络编程 Socket 我们开发的网络应用程序位于应用层,TCP和UDP属于传输层协议,在应用层如何使用传输层的服务呢?在应用层和传输层之间,则使用套接字Socket来进行分离 套接字就像是传输层为应用层开的一个小口,应用程…...

思科(Cisco ASA/Firepower)、华三(H3C)、华为(Huawei USG)防火墙 的基础配置
以下是针对 思科(Cisco ASA/Firepower)、华三(H3C)、华为(Huawei USG)防火墙 的基础配置指南,涵盖 区域划分、安全策略、NAT、路由 等核心功能。配置示例基于通用场景,实际部署时需根…...
华为海思系列----昇腾张量编译器(ATC)模型转换工具----入门级使用指南(LINUX版)
由于官方SDK比较冗余且经常跨文档讲解且SDK整理的乱七八糟,对于新手来说全部看完上手成本较高,本文旨在以简短的方式介绍 CAFFE / ONNX 模型转 om 模型,并进行推理的全流程。希望能够帮助到第一次接触华为海思框架的道友们。大佬们就没必要看这种基础文章啦! 注:本…...
supabase 怎么新建项目?
在 Supabase 中新建项目主要通过官方网站的仪表盘 (Dashboard) 来完成。以下是详细步骤: 通过 Supabase 仪表盘新建项目: 注册/登录 Supabase 账户: 访问 Supabase 官网:https://supabase.com/如果你还没有账户,点击 …...

Windows环境下maven的安装与配置
1.检查JAVA_HOME环境变量 Maven是使用java开发的,所以必须知道当前系统环境中的JDK的安装目录。 搜索栏直接输入“cmd” 或者 WinR 输入cmd 在打开的终端窗口输入“echo %JAVA_HOME”,就可以看到jdk的位置了。 如果没有的话,请参考我的文章&a…...