首页 > 其他分享 >trie(字典树)学习笔记

trie(字典树)学习笔记

时间:2023-11-05 17:12:41浏览次数:40  
标签:trie 31 ++ 笔记 int oplus void 字典

trie(字典树)学习笔记

trie 可以在 \(O(nL)\) 的时间, \(O(n\left| \Sigma\right|L)\) 的空间完成插入,查找字符串。其中 \(L\) 为字符串长,\(\Sigma\) 为字符集

int trie[N][26], tot;
int tag[N];
void insert() {
    int n = str.size();
    int u = 0;
    for (int i = 0; i < n; i++) {
        int x = str[i] - 'a';
        if (!trie[u][x]) trie[u][x] = ++tot; 
        u = trie[u][x];
    }
    tag[u] = 1;
}
void query() {
    int n = str.size();
    int u = 0;
    for (int i = 0; i < n; i++) {
        int x = str[i] - 'a';
        if (!trie[u][x]) {
            break;
        }    
        u = trie[u][x];
    }
    if (tag[u] == 0) {
        cout << "WRONG\n";
    }
    else if (tag[u] == 1) {
        cout << "OK\n"; 
        tag[u] = 2;
    }
    else {
        cout << "REPEAT\n";
    }
}

trie 维护异或极值

问题:最长异或路径

给定一棵 \(n\) 个点的无向带权树,结点下标从 \(1\) 开始到 \(n\)。寻找树中找两个结点,求最长的异或路径。边权 \(w\) 有

\(w < 2^{31}\)

解析:记 \(d(u, v)\) 为 \(u\to v\) 的异或路径值。

我们选择一个点为根,不妨为 \(1\) 。

那么我们有这个重要结论: \(d(1,u) \oplus d(1, v)=d(u,v)\) 这是由于 \(a\oplus a=0\) 和 \(a\oplus 0=a\), \(u, v\) 的 lca 部分被抵消了。

那么问题就转化成了 枚举每个 \(d(1, u)\) 找到其对应的 \(d(1, v)\) 使得 \(d(1,u) \oplus d(1, v)\) 最大,更新答案。

如何快速做到这一点呢?我们可以注意到一个性质:对每个数补全到 \(31\) 位后,如果 \(u\) 与 \(v\) 的最高位不同,但 \(u\) 与 \(w\) 的最高位相同,那么 \(u\oplus v\) 一定大于 \(u\oplus w\)。

基于这一点,我们可以将每个 \(d(1, u)\) 都将其视作一个 \(31\) 位的二进制数插入到 trie 中。枚举 \(d(1, u)\) 时从最高位开始采用贪心,尽可能地向不同于当前这一位的子树走,最后的结果就是最大的。

时空复杂度同上。这里 \(\Sigma\) 为 \(2\),\(L\) 为 \(31\)。

代码:

#include<bits/stdc++.h>
using namespace std;
using pii = pair<int, int>;
#define fi first
#define se second
const int N = 1e5 + 10, M = 31;
int trie[N * M][2], tot;
int dis[N];
int n, ans;
vector<pii> g[N];
void insert(int x) {
    for (int i = 30, u = 0; i >= 0; i--) {
        int c = (x >> i) & 1; // i-th bit
        if (!trie[u][c]) trie[u][c] = ++tot;
        u = trie[u][c];   
    }
}
void update(int x) {
    int res = 0;
    for (int i = 30, u = 0; i >= 0; i--) {
        int c = (x >> i) & 1;
        if (trie[u][!c]) {
            u = trie[u][!c];
            res += (1 << i);
        }
        else u = trie[u][c];
    }
    ans = max(ans, res);
}
void dfs(int u, int fa) {
    insert(dis[u]);
    update(dis[u]);
    for (auto x : g[u]) {
        int v = x.fi, w = x.se;
        if (v == fa) continue;
        dis[v] = dis[u] ^ w;
        dfs(v, u);
    }
}
int main() {
    cin.tie(0)->sync_with_stdio(0);
    cin >> n;
    for (int i = 0; i < n - 1; i++) {
        int u, v, w;
        cin >> u >> v >> w;
        auto add = [&] (int u, int v, int w) {
            g[u].push_back({v, w});
        };
        add(u, v, w);
        add(v, u, w);
    }
    dfs(1, 0);
    cout << ans << "\n";
    return 0;
}

标签:trie,31,++,笔记,int,oplus,void,字典
From: https://www.cnblogs.com/rhineofts/p/17810736.html

相关文章

  • 学习笔记8
    知识点归纳个人计算机定时器:指用于计算机系统中的一个工具或功能,用于设置和管理计算机系统中的定时任务或定时操作。个人计算机定时器可以用于多种用途,例如:系统定时关机:可以在一定时间后自动关闭计算机。定时提醒:可以设置定时提醒,例如定时提醒用户休息、完成某个任务等。定......
  • 学习笔记八
    学习笔记八一、作业要求自学教材第5章,提交学习笔记(10分),评分标准如下知识点归纳以及自己最有收获的内容,选择至少2个知识点利用chatgpt等工具进行苏格拉底挑战,并提交过程截图,提示过程参考下面内容(4分)"我在学*X知识点,请你以苏格拉底的方式对我进行提问,一次一个问题"**核心是......
  • 《信息安全系统设计与实现》第八次学习笔记
    第五章:定时器及时钟服务硬件定时器定时器是由时钟源和可编程计数器组成的硬件设备。时钟源通常是一个晶体振荡器,会产生周期性电信号,以精确的频率驱动计数器。使用一个倒计时值对计数器进行编程,每个时钟信号减1。当计数减为0时,计数器向CPU生成一个定时器中断,将计数值重新加载到......
  • mit6.828 - lab3练习笔记
    PartAExercise1练习1.修改`kern/pmap.c`中的`mem_init()`,分配并映射`envs`数组。该数组由`Env`结构的`NENV`实例组成,分配方式与分配页面数组类似。与页面数组一样,支持`envs`的内存也应在`UENVS`(定义于`inc/mlayout.h`)处映射为用户只读,这样用户进程才能读取该......
  • 《信息安全系统设计与实现》第九周学习笔记
    硬件定时器定时器是由时钟源和可编程计数器组成的硬件设备。时钟源通常是一个晶体振荡器,会产生周期性电信号,以精确的频率驱动计数器。使用一个倒计时值对计数器进行编程,每个时钟信号减1。当计数减为0时,计数器向CPU生成一个定时器中断,将计数值重新加载到计数器中,并重复倒计时。......
  • 信息安全系统设计与实现学习笔记8
    学习笔记8-重点总结1.定时器及时钟服务1.1硬件定时器由时钟源和可编程计数器组成的硬件设备。时钟源通常是晶体振荡器,驱动计数器以精确的频率。计数器周期称为定时器刻度,是系统的基本计时单元。1.2个人计算机定时器实时时钟(RTC)提供时间和日期信息,即使在关机时也能......
  • yzy第八周学习笔记
    定时器及时钟服务硬件定时器定时器是由时钟源和可编程计数器组成的硬件设备。时钟源通常是一个晶体振荡器,会产生周期性电信号,以精确的频率驱动计数器。使用一个倒计时值对计数器进行编程,每个时钟信号减1.当计数减为0时,计数器向CPU生成一个定时器中断,将计数值重新加载到计数器中,......
  • 大总结:uboot复习--Apple的学习笔记
    一,前言发现现在的uboot做的越来像linux驱动了,包括了设备树及其驱动模型。所以若复习设备树的话,在linux上学习和在uboot上学习是一样的,再加上我学习过了qemu仿真,所以想找到单步仿真调试方法。主要是am335x的调试器当时我焊接失败,所以只考虑仿真,另外发现stm32F407也有uboot支持,所以研......
  • 开发板nfs挂载桥接虚拟机的文件系统环境搭建--Apple的学习笔记
    一,前言我之前虚拟机配置的是NAT方式,不是桥接,然后Kernel及uboot都同nfs挂载。所以先改成了最简单的桥接方式的虚拟机。二,ubuntu虚拟机设置1,vmware先设置为桥接。2,设置ubuntu14.04的静态ip地址gedit/etc/network/interfaces内容autoeth0ifaceeth0inetstaticaddress192.168.7.......
  • 开发板nfs挂载NAT虚拟机的文件系统环境搭建--Apple的学习笔记
    一,前言总体来说我还是想用NAT虚拟机,所以基于开发板nfs挂载桥接虚拟机的文件系统环境搭建--Apple的学习笔记中的配置继续修改。二,ubuntu虚拟机中nfs挂载设置修改ip地址为192.168.112.11添加路由端口sudogedit/etc/services最后添加mountd9999/tcpmountd9999/udpPC以太网2设......