字典树(Trie树)详解

字典树(Trie树)详解

理论模块:

trie 树

字典树是一种用于实现字符串快速检索的多叉树结构

trie 的每个节点都拥有若干个字符指针,若在插入或检索字符串时扫描到一个字符 c c c

就沿着当前节点的 c c c 字符指针,走向该指针指向的节点

下图即为一个简易版字典树,存储了单词:abcbacabd

实现

(1) 初始化

一颗空 trie 仅包含一个根节点,该字符的指针均指向空

(2) 插入

当需要插入一个字符串 s s s 时,我们令一个指针 P P P 起初指向根节点。然后依次扫描 s s s 中的每一个字符 c c c

  • 1.若 P P P c c c 字符指针指向一个已经存在的节点 Q Q Q ,则令 P = Q P=Q P=Q

  • 2.若 P P P c c c 字符指针指向空,则新建一个节点 Q Q Q P P P c c c 字符指针指向 Q Q Q ,然后令 P = Q P=Q P=Q

  • S S S 中的字符扫描完毕,在当前节点 P P P 上标记它是一个字符串的末尾

(3) 检索

当需要检索一个字符串 S S S T r i e Trie Trie 中是否存在时,我们令一个指针 P P P 起初指向根节点,然后依次扫描 S S S 中的每个字符 c c c

  • (1) P P P c c c 字符指针指向空,则说明 S S S 没有被插入到 T r i e Trie Trie 树中,结束检索

  • (2) P P P c c c 字符指针指向一个已经存在的节点 Q Q Q ,则令 P = Q P=Q P=Q

    S S S 中的字符扫描完毕

    若在当前节点 P P P 被标记为一个 字符串的末尾,则说明 S 在 Trie 树中存在,否则说明 S 没有被插入过

#include<bits/stdc++.h>
using namespace std;
int trie[ ][ ],tot=1;
char str[ ];
bool end[ ];
void put()
{//1 int len;//2int p=1;//3for(int i=1;i<=len;i++){int ch=str[i]-'a';trie[p][ch]//(1)指向空trie[p][ch]=++tot;//(2)指向已存在p=trie[p][ch]; }//4end[p]=true;return;
}
bool find()
{//1 int len;//2int p=1;//3for(int i=1;i<=len;i++){int ch=str[i]-'a';trie[p][ch]//(1)指向空return false;//(2)指向已存在p=trie[p][ch]; }//4return end[p];
}
int main(){return 0;
}

例题:luoguP8306[模板]字典树

分析:

给定 n n n 个模式串 1,2,…, s 1 s_1 s1, s 2 s_2 s2,…, s n s_n sn q 次询问,每次询问给定一个文本串 t i t_i ti,请回答 1∼ s 1 s_1 s1 s n s_n sn 中有多少个字符串 s j s_j sj 满足 t i t_i ti s j s_j sj前缀

那么很明显,这道题主要考的就是我们对于字典树的运用。

除了上文所述的存储方法,还可以运用与其他树形结构相类似的结构体存储:

struct NODE{int cnt=0;int son[65];
}node[MAXN];

不过字符还需要转换成一个数字,这里我们就需要用到一个类似于map映射的东西:

int getnumber(char ch)
{if(ch>='A'&&ch<='Z')return ch-'A';if(ch>='a'&&ch<='z')return ch-'a'+26;return ch-'0'+52;
}//约等于map映射 

插入操作:

void insert(string s)//插入操作,s表示需要插入的字符串 
{now=start;for(int i=0;i<s.size();i++){num=getnumber(s[i]);if(node[now].son[num]==0)node[now].son[num]=++cnt;node[now].cnt++;now=node[now].son[num];}node[now].cnt++;return;
}

查询操作:

void find(string s)
{now=start;for(int i=0;i<s.size();i++){num=getnumber(s[i]);if(node[now].son[num]==0){printf("0\n");return;}now=node[now].son[num];}printf("%d\n",node[now].cnt);return;
}

整个代码:

#include<bits/stdc++.h>
using namespace std;
const int MAXN=3e6+5;
int num,now,start,cnt;
int T,n,q;
string s;
int getnumber(char ch)
{if(ch>='A'&&ch<='Z')return ch-'A';if(ch>='a'&&ch<='z')return ch-'a'+26;return ch-'0'+52;
}//约等于map映射 
struct NODE{int cnt=0;int son[65];
}node[MAXN];
void insert(string s)//插入操作,s表示需要插入的字符串 
{now=start;for(int i=0;i<s.size();i++){num=getnumber(s[i]);if(node[now].son[num]==0)node[now].son[num]=++cnt;node[now].cnt++;now=node[now].son[num];}node[now].cnt++;return;
}
void find(string s)
{now=start;for(int i=0;i<s.size();i++){num=getnumber(s[i]);if(node[now].son[num]==0){printf("0\n");return;}now=node[now].son[num];}printf("%d\n",node[now].cnt);return;
}
int main(){scanf("%d",&T);while(T--){scanf("%d%d",&n,&q);while(n--){cin >> s;insert(s);}while(q--){cin >> s;find(s);}start=++cnt;}return 0;
}
AC记录

如果以上内容看完还有不明白的,可以去看看博主发的这个讲解视频

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/820943.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Qt 学习笔记】QWidget的windowOpacity属性 | cursor属性 | font属性

博客主页&#xff1a;Duck Bro 博客主页系列专栏&#xff1a;Qt 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ QWidget的windowOpacity属性 | cursor属性 | font属性 文章编号&#…

leetcode hot100_day20

4/14/2024 128.最长连续序列 自己的 这是前两天做一半的题目了。这题给我的教训就是用哈希表的时候一定一定要考虑重复元素的问题&#xff01;&#xff01;&#xff01;&#xff01; 这题让我想到了最长递增子序列&#xff0c;只是名字有点像。子序列和子数组还不一样一个连续…

算法练习第18天|111.二叉树的最小深度

111.二叉树的最小深度 111. 二叉树的最小深度 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/minimum-depth-of-binary-tree/description/ 题目描述&#xff1a; 给定一个二叉树&#xff0c;找出其最小深度。 最小深度是从根节点到最近叶子节点的最…

Unity 左右折叠显示与隐藏UI的简单实现

要实现一个简单的UI左右折叠显示与隐藏&#xff0c;可以结合遮罩&#xff0c;通过代码控制UI区块的宽度和位移来实现。 具体可以按以下步骤实现&#xff1a; 1、新建一个Image组件&#xff0c;并添加精灵&#xff0c;调整大小后&#xff0c;复制一份作为该UI的父物体&#xf…

CANoe中LIN工程主节点的配置(如何切换调度表)

1&#xff1a;前置条件 1&#xff09;工程已经建立&#xff0c;simulation窗口已经配置好&#xff08;包括且不限于通道mappin好&#xff0c;数据库文件已经添加&#xff09; 2&#xff09;我已系统自带sampleCfg工程&#xff0c;作为例子。如下图 2 &#xff1a;主节点的配置…

边缘计算网关主要有哪些功能?-天拓四方

随着物联网&#xff08;IoT&#xff09;的快速发展和普及&#xff0c;边缘计算网关已经成为了数据处理和传输的重要枢纽。作为一种集成数据采集、协议转换、数据处理、数据聚合和远程控制等多种功能的设备&#xff0c;边缘计算网关在降低网络延迟、提高数据处理效率以及减轻云数…

基于51单片机的温度控制恒温箱设计—数码管显示

基于51单片机的温度控制恒温箱 &#xff08;仿真&#xff0b;程序&#xff0b;原理图&#xff0b;PCB&#xff0b;设计报告&#xff09; 功能介绍 具体功能&#xff1a; 1.DS18B20温度传感器测温&#xff1b; 2.数码管实时显示温度&#xff1b; 3.按键设置温度上下限阈值&am…

Spring Boot 统一功能处理(二)

本篇主要介绍Spring Boot统一功能处理中的统一数据返回格式。 目录 一、定义统一的返回类 二、配置统一数据格式 三、测试配置效果 四、统一格式返回的优点 五、源码角度解析String问题 一、定义统一的返回类 在我们的接口在处理请求时&#xff0c;返回的结果可以说是参…

【CAN】采样点介绍及测试方法

文章目录 1 什么是采样点2 为什么需要采样点3 采样点的计算公式4 VH6501测试原理和方法4.1 VH6501测试采样点原理4.2 VH6501测试方法 >>返回总目录<< 1 什么是采样点 采样点是节点判断信号逻辑电平的位置&#xff0c;是CAN控制器读取总线电平&#xff0c;并解释各…

【Git教程】(十二)工作流之项目设置 — 何时使用工作流,工作流的结构,项目设置概述、执行过程及其实现 ~

Git教程 工作流之项目设置 1️⃣ 何时使用工作流2️⃣ 工作流的结构3️⃣ 概述4️⃣ 使用要求5️⃣ 执行过程及其实现5.1 基于项目目录创建一个新的版本库5.2 以文件访问的方式共享版本库5.3 用 Git daemon 来共享版本库5.4 用 HTTP 协议来共享版本库5.5 用 SSH 协议来共享版…

【论文阅读02】一种基于双通道的水下图像增强卷积神经网络

来源&#xff1a;海洋论坛▏一种基于双通道的水下图像增强卷积神经网络 当前不会的 一、背景&#xff1a; 水下图像增强方法包含有无水下成像模型的水下图像增强方法、基于水下成像模型的水下图像恢复方法、水下成像模型与深度学习相结合的方法以及完全采用深度学习的方…

基于Python的景区票务人脸识别系统(V2.0)

博主介绍&#xff1a;✌IT徐师兄、7年大厂程序员经历。全网粉丝15W、csdn博客专家、掘金/华为云//InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;&#x1f3…

数据结构速成--数据结构和算法

由于是速成专题&#xff0c;因此内容不会十分全面&#xff0c;只会涵盖考试重点&#xff0c;各学校课程要求不同 &#xff0c;大家可以按照考纲复习&#xff0c;不全面的内容&#xff0c;可以看一下小编主页数据结构初阶的内容&#xff0c;找到对应专题详细学习一下。 目录 一…

【Linux】磁盘阵列RAID技术

目录 一、RAID介绍 1.1 什么是RAID技术&#xff1f; 1.2 为什么要使用RAID技术&#xff1f; 二、RAID级别 2.1 常见的RAID级别 2.2 常见RAID介绍 三、RAID特性对比 一、RAID介绍 1.1 什么是RAID技术&#xff1f; 把多块独立的物理磁盘按不同的方式组合起来形成一个硬盘…

【ENSP】华为三层交换机配置AAA认证,开启telnet服务

配置步骤 1.给交换机配置ip地址&#xff0c;以便登陆 2.配置AAA&#xff0c;用户名&#xff0c;密码&#xff0c;服务类型&#xff0c;用户权限 3.配置接入设备的数量 4.开启telnet服务 LSW2交换机配置 u t m #关闭提示 sys …

基于单链表实现通讯管理系统!(有完整源码!)

​ 个人主页&#xff1a;秋风起&#xff0c;再归来~ 文章专栏&#xff1a;C语言实战项目 个人格言&#xff1a;悟已往之不谏&#xff0c;知来者犹可追 克心守己&#xff0c;律己则安&#xff01; 1、前言 友友们&#xff0c;这篇文章是基于单链…

使用Python模仿文件行为

在Python中&#xff0c;你可以通过文件操作函数&#xff08;如open()函数&#xff09;以及模拟输入输出流的库&#xff08;如io模块&#xff09;来模拟文件行为。下面是一些示例&#xff0c;展示了如何使用这些工具在Python中模拟文件行为。 1、问题背景 在编写一个脚本时&…

Springboot框架——4.整合jdbc

1.pom.xml中导入依赖&#xff1a; <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jdbc</artifactId> </dependency> <dependency><groupId>org.springframework.boot</groupI…

SpringMVC(三)【REST 风格】

1、REST 风格 1.1、REST 简介 REST&#xff08;Representational State Transfer&#xff09;&#xff0c;表现形式状态转换 在开发中&#xff0c;它其实指的就是访问网络资源的格式 1.1.1、传统风格资源描述形式 http://localhost/user/getById?id1http://localhost/user…