当前位置: 首页 > news >正文

【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)结构,支持插入、搜索和前缀匹配功能。以下是对代码的详细分析和优化建议:


代码思路分析

  1. 数据结构设计

    • isEnd:标记当前节点是否为单词的结尾。
    • next[26]:一个包含 26 个指针的数组,对应英文小写字母,用于构建字符到子节点的映射。
  2. 核心方法

    • 插入 (insert): 从根节点开始,逐字符向下遍历。若字符对应的子节点不存在,则创建新节点。遍历完成后,标记最后一个节点为单词结尾。
    • 搜索 (search): 检查单词是否存在且最后一个节点被标记为结尾。
    • 前缀匹配 (startsWith): 仅检查路径是否存在,无需判断结尾标记。
  3. 正确性

    • 逻辑正确,符合前缀树的标准实现。
    • 使用 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;}
};

相关文章:

  • 基于C#+MySQL实现(WinForm)企业设备使用信息管理系统
  • niushop单商户V5多门店版V5.5.0全插件+商品称重、商家手机端+搭建环境教程
  • 从数据中台到数据飞轮:数字化转型的演进之路
  • python如何做人脸识别
  • Docker与PostgreSQL
  • 国联股份卫多多与七腾机器人签署战略合作协议
  • 基于Spring Boot+Layui构建企业级电子招投标系统实战指南
  • 工业巡检机器人 —— 机器人市场的新兴增长引擎
  • OpenTiny icons——超轻量的CSS图标库,引领图标库新风向
  • Git 第二讲---提高篇 git的分支管理
  • AD 飞线的显示与隐藏
  • 图论part09dijkstra算法
  • 【时时三省】(C语言基础)使用字符串处理函数
  • 四旋翼控制入门笔记
  • 国内USB IP商业解决方案新选择:硬件USB Server
  • [中国版 Cursor ]?!CodeBuddy快捷搭建个人展示页面指南
  • 在SQL Server中调整查询超时j解决方案
  • 大模型交叉研讨课笔记-Transformer架构
  • 化工行业专利管理系统:全流程解决方案解析
  • React Native 与 Expo
  • 台行政机构网站删除“汉人”改为“其余人口”,国台办回应
  • 国务院关税税则委:调整对原产于美国的进口商品加征关税措施
  • 讲一个香港儿童的故事,《劏房的天空》获“周庄杯”特等奖
  • 专访|韩国世宗研究所中国研究中心主任:李在明若上台将推行均衡外交
  • 《AI×SCIENCE十大前沿观察》9:合成数据和数据基础设施
  • 梅花奖在上海|“我的乱弹我的团”,民营院团首次入围终评