【算法基础:数据结构】2.2 字典树/前缀树 Trie

文章目录

  • 知识点
    • cpp结构体模板
  • 模板例题
    • 835. Trie字符串统计❤️❤️❤️❤️❤️(重要!模板!)
    • 143. 最大异或对😭😭😭😭😭(Trie树的应用)
  • 相关题目练习
    • 208. 实现 Trie (前缀树)
    • 1804. 实现 Trie (前缀树) II
  • 参考资料

知识点

用于高效地存储和查找字符串集合的数据结构——Trie树

https://oi-wiki.org/string/trie/
在这里插入图片描述
可以发现,这棵字典树用边来代表字母,而从根结点到树上某一结点的路径就代表了一个字符串。举个例子, 1 → 4 → 8 → 12 1\to4\to 8\to 12 14812 表示的就是字符串 caa

这类题目,字母的种类不会很多。

cpp结构体模板

struct trie {int nex[100000][26], cnt;bool exist[100000];  // 该结点结尾的字符串是否存在void insert(char *s, int l) {  // 插入字符串int p = 0;for (int i = 0; i < l; i++) {int c = s[i] - 'a';if (!nex[p][c]) nex[p][c] = ++cnt;  // 如果没有,就添加结点p = nex[p][c];}exist[p] = 1;}bool find(char *s, int l) {  // 查找字符串int p = 0;for (int i = 0; i < l; i++) {int c = s[i] - 'a';if (!nex[p][c]) return 0;p = nex[p][c];}return exist[p];}
};

模板例题

835. Trie字符串统计❤️❤️❤️❤️❤️(重要!模板!)

https://www.acwing.com/activity/content/problem/content/883/

在这里插入图片描述

代码模板在于 insertquery 这两个方法的写法。

除此之外要理解数组 soncnt 和变量 idx 的含义。(含义已经写在代码注释里了

son[][] 的第一维是可能出现的字符**数量**的最大值;第二维是可能出现的字符**种类**的最大值。
cnt[] 的大小是可能出现的字符数量的最大值,也就是记录每个节点作为了几次字符串的末尾。
idx 记录出现了几个新的节点。
import java.io.BufferedInputStream;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.util.*;public class Main {static final int N = 100010;    // 所有输入的字符串总长度不超过 10^5static int[][] son;static int[] cnt;static int idx;             // idx递增作为节点的序号static {son = new int[N][26];   // 记录各个节点的儿子cnt = new int[N];       // 记录各个节点作为结尾的次数}public static void main(String[] args) throws IOException {Scanner sin = new Scanner(new BufferedInputStream(System.in));BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));int n = sin.nextInt();while (n-- > 0) {char op = sin.next().charAt(0);String s = sin.next();if (op == 'I') {insert(s.toCharArray());} else {System.out.println(query(s.toCharArray()));}}bw.flush();}// 插入一个字符串static void insert(char[] str) {int p = 0;for (int i = 0; i < str.length; ++i) {		// 枚举每个字符int u = str[i] - 'a';if (son[p][u] == 0) son[p][u] = ++idx;  // 如果当前层不存在u的话,新建一个节点p = son[p][u];}cnt[p]++;           // 作为结尾的情况+1}static int query(char[] str) {int p = 0;for (int i = 0; i < str.length; ++i) {int u = str[i] - 'a';if (son[p][u] == 0) return 0;p = son[p][u];}return cnt[p];}
}

143. 最大异或对😭😭😭😭😭(Trie树的应用)

https://www.acwing.com/problem/content/145/

在这里插入图片描述

异或运算:相同得 0 ,不同得 1。(俗称不进位加法)

从高位开始比较。

检查到有反的,就可以 += 1 << i;

import java.util.Scanner;public class Main {final static int M = 31 * 100010;   // M 是 Trie树中最多可能的节点数量static int[][] son = new int[M][2];static int idx = 0;public static void main(String[] args){Scanner scanner = new Scanner(System.in);int n = scanner.nextInt(), ans = 0;for (int i = 0; i < n; ++i) {int a = scanner.nextInt();ans = Math.max(find(a), ans);insert(a);}System.out.println(ans);}// 从高位到低位插入public static void insert(int x) {int p = 0;for (int i = 30; i >= 0; --i) {int u = x >> i & 1;if (son[p][u] == 0) son[p][u] = ++idx;p = son[p][u];}}public static int find(int x) {int p = 0, res = 0;for (int i = 30; i >= 0; --i) {int u = x >> i & 1;         // 获得当前位if (son[p][u ^ 1] != 0) {   // 检查当前位有没有取反的res += 1 << i;p = son[p][u ^ 1];} else p = son[p][u];}return res;}
}

相关题目练习

208. 实现 Trie (前缀树)

https://leetcode.cn/problems/implement-trie-prefix-tree/
在这里插入图片描述
提示:

1 <= word.length, prefix.length <= 2000
word 和 prefix 仅由小写英文字母组成
insert、search 和 startsWith 调用次数 总计 不超过 3 * 10^4 次

一道练习Trie树模板的题目。

class Trie {final int N = 200001 + 1;int[][] son = new int[N][26];int[] cnt = new int[N];int idx = 0;public Trie() {}public void insert(String word) {int p = 0;for (char ch: word.toCharArray()) {int u = ch - 'a';if (son[p][u] == 0) son[p][u] = ++idx;p = son[p][u];}++cnt[p];}public boolean search(String word) {int p = 0;for (char ch: word.toCharArray()) {int u = ch - 'a';if (son[p][u] == 0) return false;p = son[p][u];}return cnt[p] > 0;}public boolean startsWith(String prefix) {int p = 0;for (char ch: prefix.toCharArray()) {int u = ch - 'a';if (son[p][u] == 0) return false;p = son[p][u];}return true;}
}/*** Your Trie object will be instantiated and called as such:* Trie obj = new Trie();* obj.insert(word);* boolean param_2 = obj.search(word);* boolean param_3 = obj.startsWith(prefix);*/ 

1804. 实现 Trie (前缀树) II

1804. 实现 Trie (前缀树) II

在这里插入图片描述
提示:

1 <= word.length, prefix.length <= 2000
word 和 prefix 只包含小写英文字母。
insert、 countWordsEqualTo、 countWordsStartingWith 和 erase 总共调用最多 3 * 10^4 次。
保证每次调用 erase 时,字符串 word 总是存在于前缀树中。

相比上一题,多开一个数组 cnt2 记录一下各个节点被经过了多少次就好了

class Trie {int[][] son = new int[30000][26];int[] cnt = new int[30000], cnt2 = new int[30000];int idx = 0;public Trie() {}public void insert(String word) {int p = 0;for (char ch: word.toCharArray()) {int u = ch - 'a';if (son[p][u] == 0) son[p][u] = ++idx;p = son[p][u];cnt2[p]++;}cnt[p]++;}public int countWordsEqualTo(String word) {int p = getP(word);return cnt[p];}public int countWordsStartingWith(String prefix) {int p = getP(prefix);return cnt2[p];}public void erase(String word) {int p = 0;for (char ch: word.toCharArray()) {int u = ch - 'a';p = son[p][u];cnt2[p]--;}cnt[p]--;}public int getP(String s) {int p = 0;for (char ch: s.toCharArray()) {int u = ch - 'a';if (son[p][u] != 0) p = son[p][u];else return 0;}return p;}
}/*** Your Trie object will be instantiated and called as such:* Trie obj = new Trie();* obj.insert(word);* int param_2 = obj.countWordsEqualTo(word);* int param_3 = obj.countWordsStartingWith(prefix);* obj.erase(word);*/

参考资料

https://oi-wiki.org/string/trie/

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

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

相关文章

C# MVC 多图片上传预览

一.效果图&#xff1a; 开发框架&#xff1a;MVC&#xff0c;Layui 列表主界面这里就不展示了&#xff0c;可以去看看这篇文章&#xff1a;Layui项目实战&#xff0c;这里讲的是“上传Banner”界面功能&#xff1a; 其中包括&#xff0c;多文件上传&#xff0c;预览&#xff0c…

vue进阶-消息的订阅与发布

&#x1f4d6;vue基础学习-组件 介绍了嵌套组件间父子组件通过 props 属性进行传参。子组件传递数据给父组件通过 $emit() 返回自定义事件&#xff0c;父组件调用自定义事件接收子组件返回参数。 &#x1f4d6;vue进阶-vue-route 介绍了路由组件传参&#xff0c;两种方式&…

【conan】本地编译三方库,上传conan服务器

1.6 conan 远程已经编译好的库 conan中文博客&#xff1a; 三方库资源&#xff1a; github conan-io 本地查询 conan search Existing package recipes:b2/4.9.6 boost/1.71.0nolovr/stable bzip2/1.0.8 ceres-solver/2.0.0nolovr/stable eigen/3.3.7nolovr/stable eigen_c…

【软件测试】selenium中元素的定位

1.元素的定位 不管用那种方式&#xff0c;必须保证页面上该属性的唯一性 1.CSS 定位 CSS(Cascading Style Sheets)是一种语言&#xff0c;它被用来描述HTML 和XML 文档的表现。 CSS 使用选择器来为页面元素绑定属性。这些选择器可以被selenium 用作另外的定位策略CSS的获取可…

C++基础算法前缀和和差分篇

&#x1f4df;作者主页&#xff1a;慢热的陕西人 &#x1f334;专栏链接&#xff1a;C算法 &#x1f4e3;欢迎各位大佬&#x1f44d;点赞&#x1f525;关注&#x1f693;收藏&#xff0c;&#x1f349;留言 主要讲解了前缀和和差分算法 文章目录 Ⅳ. 前缀和 和 差分Ⅵ .Ⅰ前缀和…

echarts_柱状图+漏斗图

目录 柱状图(bar)需求[1] 复制案例[2] 修改类目轴方向[3] 修改数据渲染方向[4] 修改坐标轴文本样式 漏斗图(funnel)漏斗图的形状 柱状图(bar) 需求 如上图&#xff0c;做一个横向柱状图&#xff0c;后端返回的数据是从小向大排列的数据&#xff0c;希望能够按照顺序进行展示。…

【Docker】详解docker安装及使用

详解docker安装及使用 1. 安装docker1.1 查看docker版本信息 2. Docker镜像操作3. Docker容器操作4.知识点总结4.1 docker镜像操作4.2 docker容器操作4.3 docker run启动过程 参见docker基础知识点详解 1. 安装docker 目前Docker只能支持64位系统。 ###关闭和禁止防火墙开机自…

pytorch+CRNN实现

最近接触了一个仪表盘识别的项目&#xff0c;简单调研以后发现可以用CRNN来做。但是手边缺少仪表盘数据集&#xff0c;就先用ICDAR2013试了一下。 结果遇到了一系列坑。为了不使读者和自己在以后的日子继续遭罪。我把正确的代码发到下面了。 1&#xff09;超参数请不要调整&am…

Android oom_adj 详细解读

源码基于&#xff1a;Android R 0. 前言 在博文《oom_adj 内存水位算法剖析》一文中详细的分析了lmkd 中针对 oom_adj 内存水位的计算、使用方法&#xff0c;在博文《oom_adj 更新原理(1)》、《oom_adj 更新原理(2)》中对Android 系统中 oom_adj 的更新原理进行了详细的剖析。…

Centos 7 安装 Oracle 11G

Oracle 11G 安装教程 准备环境 p13390677_112040_Linux-x86-64_1of7.zipp13390677_112040_Linux-x86-64_2of7.zipCentos 7- rhel7-英文版的系统–不想换语言的执行(LANGen_US)– 传输 文件到服务器上 创建用户和组 [rootlocalhost ~]# groupadd oracle [rootlocalhost ~]…

Windows11 C盘瘦身

1.符号链接 将大文件夹移动到其他盘&#xff0c;创建成符号链接 2.修改Android Studio路径设置 1.SDK路径 2.Gradle路径 3.模拟器路径 设置环境变量 ANDROID_SDK_HOME

基于单片机的盲人导航智能拐杖老人防丢防摔倒发短息定位

功能介绍 以STM32单片机作为主控系统&#xff1b; OLED液晶当前实时距离&#xff0c;安全距离&#xff0c;当前经纬度信息&#xff1b;超声波检测小于设置的安全距离&#xff0c;蜂鸣器报警提示&#xff1a;低于安全距离&#xff01;超声波检测当前障碍物距离&#xff0c;GPS进…

python发送邮件yagmail库

yagmail库发送邮件简洁&#xff0c;代码量少 import yagmaildef send_yagmail(sender, send_password, addressee, hostsmtp.qq.com, port465):yag yagmail.SMTP(sender, send_password, host, port)img_url https://img2.baidu.com/it/u483398814,2966849709&fm253&…

基于单片机的智能空调系统的设计与实现

功能介绍 以51单片机作为主控系统&#xff1b;LCD1602液晶显示当前水温&#xff0c;定时提醒&#xff0c;水量变化DS18B20检测当前水体温度&#xff1b;水位传感器检测当前水位&#xff1b;继电器驱动加热片进行水温加热&#xff1b;定时提醒喝水&#xff0c;蜂鸣器报警&#x…

LeetCode面试题02.07.链表相交

面试题02.07.链表相交 两种解题思路 面试题02.07.链表相交一、双指针二、哈希集合 一、双指针 这道题简单来说&#xff0c;就是求两个链表交点节点的指针 这里注意&#xff1a;交点不是数值相等&#xff0c;而是指针相等 为了方便举例&#xff0c;假设节点元素数值相等&…

用Python采用Modbus-Tcp的方式读取485电子水尺数据

README.TXT 2023/6/15 V1.0 实现了单个点位数据通信、数据解析、数据存储 2023/6/17 V2.0 实现了多个点位数据通信、数据解析、数据存储 2023/6/19 V2.1 完善log存储&#xff0c;仅保留近3天的log记录&#xff0c;避免不必要的存储&#xff1b;限制log大小&#xff0c;2MB。架…

数字原生时代,奥哲如何让企业都成为“原住民”?

22年前&#xff0c;美国教育学家马克‧普伦斯基&#xff08;Marc Prensky&#xff09;出版了《数字原生与数字移民》&#xff08;Digital Natives, Digital Immigrants&#xff09;一书&#xff0c;首次提出了“数字原住民”和“数字移民”两大概念&#xff0c;用来定义跨时代的…

【数据结构】_1.集合与复杂度

目录 1. 集合框架 2. 时间复杂度 2.1 时间复杂度和空间复杂度 2.2 时间复杂度的概念 2.3 大O的渐进表示法 2.3.1 精确的时间复杂度表达式 2.3.2 大O渐进表示法的三条规则 2.3.3 时间复杂度的最好、平均与最坏情况 2.4 时间复杂度计算示例 3.空间复杂度 1. 集合框架 …

字节跳动后端面试,笔试部分

var code "7022f444-ded0-477c-9afe-26812ca8e7cb" 背景 笔者在刷B站的时候&#xff0c;看到了一个关于面试的实录&#xff0c;前半段是八股文&#xff0c;后半段是笔试部分&#xff0c;感觉笔试部分的题目还是挺有意思的&#xff0c;特此记录一下。 笔试部分 问…

【多线程例题】顺序打印abc线程

顺序打印-进阶版 方法一&#xff1a;三个线程竞争同一个锁&#xff0c;通过count判断是否打印 方法二&#xff1a;三个线程同时start&#xff0c;分别上锁&#xff0c;从a开始&#xff0c;打印后唤醒b 三个线程分别打印A&#xff0c;B&#xff0c;C 方法一&#xff1a;通过co…