首页 > 其他分享 >学习笔记-字典树

学习笔记-字典树

时间:2023-11-30 17:38:06浏览次数:38  
标签:ch int sum 笔记 学习 ++ 异或 oplus 字典

字典树一般有两个作用(我学到的),一个是查询单词的出现,一个是计算最大异或值。

字典树的ch数组该如何理解?
其实ch[p][j]指的是从p是否有一条值为为j的边到下一个点,如果ch[p][j]为0,就是没有。

例题1
luogu P2580

https://www.luogu.com.cn/problem/P2580

这题就是存字串的裸题,唯一要处理的细节就是点名两次该如何处理,我这里用了一个有点蠢的方法,记录了每个字符串点的次数,以及是否点过,如果次数大于2并且再一次点到就是重复点名。还有这里的cnt数组记录的是每个字符串结尾的次数。有些题种记录的是每个字符串出现的次数。

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
const int mod = 998244353, INF = 1 << 30;
const int N = 5e5 + 10;
int ch[N][26], idx, cnt[N];
map<string, bool> vis;
map<string, int> hv;


void insert(string &s)
{
    int len = s.size();
    int p = 0;
    for (int i = 0; i < len; i ++){
        int j = s[i] - 'a';
        if (!ch[p][j]) ch[p][j] = ++ idx;
        p = ch[p][j];    
    }
    cnt[p] ++;
}

int query(string &s)
{
    int len = s.size();
    int p = 0;
    for (int i = 0; i < len; i ++){
        int j = s[i] - 'a';
        if (!ch[p][j]) return 0;
        p = ch[p][j];
    }
    return cnt[p];
}

int main()
{
    cin.tie(nullptr);
    ios::sync_with_stdio(false);
    int n;
    cin >> n;
    for (int i = 1; i <= n; i ++){
        string s;
        cin >> s;
        hv[s] ++;
    }
    int q;
    cin >> q;
    while (q --){
        string s;
        cin >> s;
        if (hv[s] && !vis[s]) {cout << "OK" << endl; vis[s] = 1;}
        else if (hv[s] && vis[s]) cout << "REPEAT" << endl;
        else cout << "WRONG" << endl;
    }
    return 0;
}

思考: 我们有插入操作,那如何实现删除操作?
其实我们可以用懒删除,用一个cnt计数,这里的cnt和例1种的cnt不太一样,这个是每个节点都要计数一次,例1中是每个单词的尾部节点计数一次,删除时将每个节点的cnt减去一次,查询的时候如果cnt为0就说明节点已经删除掉了。

例题2
luogu P4551

https://www.luogu.com.cn/problem/P4551

这题是求最大的异或值,只是转移到树上,我们假设异或值最大的路径是u -> v,根据异或前缀和的思想,我们设点x到根节点的异或路径和是sum[x], u, v到最近公共祖先的异或值为s[u], s[v], s[u] ^ s[v]就是这题的答案。
s[u] ^ s[v] = s[u] ^ s[v] ^ sum[lca[u, v]] ^ sum[lca[u, v]] = (s[u] ^ sum[lca(u, v)]) ^ (s[v] ^ sum[lca(u, v)]) = sum[u] ^ sum[v]
所以我们只需要一次dfs就能算出所有节点到根节点的异或路径和,并查询所有异或路径和的最大异或值,用01trie维护。

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
const int mod = 998244353, INF = 1 << 30;
const int N = 1e5 + 10;
vector<PII> e[N];
int ch[N * 31][2], idx, sum[N];
int ans;

void insert(int x)
{
    int p = 0;
    for (int i = 30; i >= 0; i --){
        int j = (x >> i) & 1;
        if (!ch[p][j]) ch[p][j] = ++ idx;
        p = ch[p][j];
    }
}

int query(int x)
{
    int p = 0, res = 0;
    for (int i = 30; i >= 0; i --){
        int j = (x >> i) & 1;
        if (ch[p][!j]){
            res += (1 << i);
            p = ch[p][!j];
        }
        else{
            p = ch[p][j];
        }
    }
    return res;
}

void dfs(int u, int fa)
{
    for (auto x: e[u]){
        int v = x.first, w = x.second;
        if (v == fa) continue;
        sum[v] = sum[u] ^ w;
        dfs(v, u);
    }
}


int main()
{
    cin.tie(nullptr);
    ios::sync_with_stdio(false);
    int n;
    cin >> n;
    for (int i = 1; i < n; i ++) {
        int u, v, w;
        cin >> u >> v >> w;
        e[u].push_back({v, w});
        e[v].push_back({u, w});
    }
    dfs(1, -1);
    for (int i = 2; i <= n; i ++) insert(sum[i]);

    for (int i = 1; i <= n; i ++) ans = max(query(sum[i]), ans);
    cout << ans << endl;
    return 0;
}

例题3
CF1895D XOR Construction

https://codeforces.com/contest/1895/problem/D

题意:给出一个n-1长度的数组a,要构造一个b数组,满足要求:1.数组b是0 ~ n-1的排列 2.b[i] ^ b[i + 1] = a[i]

手推一下,可以发现 \(a_{1} \oplus a_{2} \oplus .... \oplus a_{n - 1} = b_{1} \oplus b_{2} \oplus b_{2} \oplus b_{3} \oplus .... \oplus b_{n - 1} \oplus b_{n}\)
即\(a_{1} \oplus a_{2} \oplus .... \oplus a_{n - 1} = b_{1} \oplus b_{n}\),可以看出我们只要求出\(b_{1}\)的值,b就能求出来。
我们设a的异或前缀和为sum[n];
我们发现题目中说保证有解,而且解的个数可能不为1。
用a的异或前缀和建一颗字典树,我们的所有值都是0 ~ n - 1的。
所以我们枚举\(a_{1}\) 从[0, n - 1], 如果与字典树中的最大异或和刚好是n - 1,这个\(a_{1}\)就是符合要求的。

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
const int mod = 998244353, INF = 1 << 30;
const int N = 2e5 + 10;
int a[N], sum[N];
int ch[N * 31][2], idx;

void insert(int x)
{
    int p = 0;
    for (int i = 30; i >= 0; i --){
        int j = (x >> i) & 1;
        if (!ch[p][j]) ch[p][j] = ++ idx;
        p = ch[p][j];
    }
}


int query(int x)
{
    int p = 0, res = 0;
    for (int i = 30; i >= 0; i --){
        int j = (x >> i) & 1;
        if (ch[p][!j]){
            res += (1 << i);
            p = ch[p][!j];
        }
        else{
            p = ch[p][j];
        }
    }
    return res;
}

int main()
{
    cin.tie(nullptr);
    ios::sync_with_stdio(false);
    int n;
    cin >> n;
    for (int i = 1; i < n; i ++){
        cin >> a[i];
        sum[i] = a[i];
        sum[i] ^= sum[i - 1];
    }
    for (int i = 1; i <= n; i ++) insert(sum[i]);

    int fst = 0;
    for (int i = 1; i <= n; i ++){
        if (query(sum[i]) == n - 1) {
            fst = i;
            break;
        }
    }
    cout << sum[fst] << ' ';
    for (int i = 1; i < n; i ++){
        cout << (sum[fst] ^ sum[i]) << ' ';
    }
    return 0;
}

标签:ch,int,sum,笔记,学习,++,异或,oplus,字典
From: https://www.cnblogs.com/lu1no/p/17867862.html

相关文章

  • Oracle 高低水位线的学习
    Oracle高低水位线的学习背景最近产品的一些脚本会大量的给一些流程表里面插入数据因为只是一个流程相关没有时序查询的需求所以数据量挺大,但是按照石时间戳删除非常麻烦.自己执行过多次delete但是使用自己的SQL查询表大小,发现总是失败想起来可能跟高低水位线有关系,......
  • 秦疆的Java课程笔记:48 方法 命令行传递参数
    一般简称“命令行传参”,了解即可。有时候需要运行一个程序时再传递给它消息。这要靠传递命令行参数给main()函数来实现。格式如下:publicclassCommandLine{ publicstaticvoidmain(Stringargs[]){ for(inti=0;i<args.length;i++){ System.out.println("a......
  • 秦疆的Java课程笔记:49 方法 可变参数
    也叫做“不定项参数”。JDK1.5开始,Java支持传递同类型的可变参数给一个方法。在方法声明中,在指定参数类型后加一个省略号(也就是三个句号)...。一个方法中只能指定一个可变参数,它必须是方法的最后一个参数。任何普通的参数必须在它之前声明。publicclassDemo1{publ......
  • 学习笔记12(PHP MySQL数据库系统)
    一、知识点梳理(一)使用PHP连接到MySQL服务器安装必要的软件:在基于Ubuntu的系统上,可以使用以下命令:sudoapt-getinstallphpmysql-serverphp-mysql启动MySQL服务:使用以下命令:sudoservicemysqlstart创建MySQL数据库和用户:登录MySQL并创建一个数据库以及一个......
  • 秦疆的Java课程笔记:50 方法 递归讲解
    一般情况下,我们用A方法调用B方法。递归就是,A方法调用A方法,自己调用自己。利用递归可以用简单的程序来解决一些复杂的问题。通常把一个大型复杂的问题转化为一个与原问题相似的规模较小的问题来求解,递归策略只需要少量的程序就可描述出解题过程所需要的多此重复计算,大大减少了程......
  • [TS手册学习] 04_对象类型
    TS官方手册:TypeScript:Handbook-TheTypeScriptHandbook(typescriptlang.org)匿名与具名对象类型的声明可以是匿名的,也可以使用interface或type进行具名声明。functiongreet(person:{name:string;age:number}){return"Hello"+person.name;}interface......
  • 《软件工程导论》阅读笔记
        软件工程导论,我认识到为解决“软件危机”引发的一系列困境,使得“软件工程”这一概念面世,其中,软件工程中由“对象+类+继承+消息”组成的面向对象的开发方法是十分重要的。软件开发的生命周期中,问题定义、可行性、需求分析、概要设计、详细设计、程序设计、测试文档、技......
  • 信息安全系统设计与实现 学习笔记12
    《Unix/Linux系统编程》14章学习笔记本章重点:MySQL关系数据库系统;MySQL;如何在Linux机器上安装和运行MySQL;如何使用MySQL在命令模式和批处理模式下使用SQL脚本创建和管理数据库;如何将MySQL与C编程相结合;如何将MySQL与PHP集成,通过动态Web页面创建和管理数据库。MySQL简介MySQL是......
  • 学习笔记12
    教材知识点总结MySQL是一个流行的开源关系型数据库管理系统(RDBMS),常用于各种Web应用程序的后端数据存储。它提供了高性能、可靠性和易用性,并且在Unix/Linux系统编程中被广泛使用。下面是对MySQL及其在Unix/Linux系统编程中的一些关键知识点的详细总结:MySQL简介MySQL是一个开源......
  • 第十三章学习笔记
    引言本章论述了TCP/IP和网络编程,分为两个部分。第一部分论述了TCP/IP协议及其应用,具体包括TCP/IP栈、IP地址、主机名、DNS、IP数据包和路由器;介绍了TCP/IP网络中的UDP和TCP协议、端口号和数据流;阐述了服务器-客户机计算模型和套接字编程接口;通过使用UDP和TCP套接字的示......