[学习笔记]我们追过的神奇异或(Trie树系列)
引言
刚学了\(Trie\)树,写篇博客巩固一下。
题目
首先安利一发\(Trie\)树模板
第一题不难,但是我用的是暴力。
构造一棵\(Trie\)树,然后把字符串长度从大到小排序,然后按顺序插入。若在插入的时候\(Trie\)树结点一直不为空,那么该串为其的子串。
\(Code\ Below:\)
#include <bits/stdc++.h>
using namespace std;
int n,trie[100010][10],ans,tot;
char s[100010][15];
struct node{
int len,id;
}l[100010];
void insert(int k){
int len=strlen(s[k]),p=0,b=0;
for(int i=0;i<len;i++){
if(trie[p][s[k][i]-'0']==-1) {
trie[p][s[k][i]-'0']=++tot;b=1;
}
p=trie[p][s[k][i]-'0'];
}
if(b==0) ans=1;
}
bool cmp(node a,node b){
return a.len>b.len;
}
int main()
{
int t;
scanf("%d",&t);
while(t--){
memset(trie,-1,sizeof(trie));
scanf("%d",&n);
ans=0;tot=0;
for(int i=1;i<=n;i++){
scanf("%s",s[i]);
l[i].len=strlen(s[i]);l[i].id=i;
}
sort(l+1,l+n+1,cmp);
for(int i=1;i<=n;i++){
if(ans==0) insert(l[i].id);
}
if(ans==1) printf("NO\n");
else printf("YES\n");
}
return 0;
}
第二题构造一棵\(Trie\)树,然后根据异或的运算规则,尽量当前的值与结点的值不一样。
此题是二三题的基础。
\(Code\ Below:\)
#include <cstdio>
int n,trie[100010*31][2],ans=0,val,tot=0;
int max(int a,int b){return a>b?a:b;}
int search(int x){
int p=0,num=0,c;
for(int i=31;i>=0;i--){
c=(x>>i)&1;
if(trie[p][c^1])
p=trie[p][c^1],num=num<<1|1;
else p=trie[p][c],num=num<<1;
}
return num;
}
void insert(int x){
int p=0,c;
for(int i=31;i>=0;i--){
c=(x>>i)&1;
if(!trie[p][c]) trie[p][c]=++tot;
p=trie[p][c];
}
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%d",&val);
ans=max(ans,search(val));
insert(val);
}
printf("%d\n",ans);
return 0;
}
定义前缀异或和\(sum[1..i]=a[1]\)$...$\(a[i]\),\(v[i..j]=a[i]\)$a[i+1]$\(...\)^\(a[j]\),则\(v[i..j]=sum[1..i] * sum[1..j]\)
由于\(a^a=0,0^a=a,\)所以异或两次值不变。
那么预处理出树上前缀异或和(即到根的前缀异或和),跑一遍\(The\ XOR\ largest\ pair\)
\(Code\ Below:\)
#include <cstdio>
int n,trie[100010*31][2],ans=0,a[100010],val,tot=0,t=0,head[100010];
int max(int a,int b){return a>b?a:b;}
struct node{
int to,val,next;
}e[100010*2];
void add(int x,int y,int w){
e[++t].to=y;
e[t].val=w;
e[t].next=head[x];
head[x]=t;
}
void dfs(int x,int fa,int now){
a[x]=now;
for(int i=head[x];i;i=e[i].next){
int y=e[i].to;
if(y==fa) continue;
dfs(y,x,now^e[i].val);
}
}
int search(int x){
int p=0,num=0,c;
for(int i=31;i>=0;i--){
c=(x>>i)&1;
if(trie[p][c^1])
p=trie[p][c^1],num=num<<1|1;
else p=trie[p][c],num=num<<1;
}
return num;
}
void insert(int x){
int p=0,c;
for(int i=31;i>=0;i--){
c=(x>>i)&1;
if(!trie[p][c]) trie[p][c]=++tot;
p=trie[p][c];
}
}
int main()
{
scanf("%d",&n);
for(int i=1;i<n;i++){
int x,y,w;
scanf("%d%d%d",&x,&y,&w);
add(x,y,w);add(y,x,w);
}
dfs(1,1,0);
for(int i=1;i<=n;i++){
ans=max(ans,search(a[i]));
insert(a[i]);
}
printf("%d\n",ans);
return 0;
}
第四题预处理出前缀异或和和后缀异或和,然后用\(O(n)\)的时间求最大值
\(Code\ Below:\)
#include <cstdio>
int n,trie[400010*31][2],ans=0,val,tot=0,a[100010],l[400010],r[400010];
int max(int a,int b){return a>b?a:b;}
int search(int x){
int p=0,num=0,c;
for(int i=31;i>=0;i--){
c=(x>>i)&1;
if(trie[p][c^1])
p=trie[p][c^1],num=num<<1|1;
else p=trie[p][c],num=num<<1;
}
return num;
}
void insert(int x){
int p=0,c;
for(int i=31;i>=0;i--){
c=(x>>i)&1;
if(!trie[p][c]) trie[p][c]=++tot;
p=trie[p][c];
}
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
scanf("%d",&a[i]);
int now=0;insert(now);
for(int i=1;i<=n;i++){
now^=a[i];
l[i]=max(l[i-1],search(now));
insert(now);
}
now=0;insert(now);
for(int i=n;i>=1;i--){
now^=a[i];
r[i]=max(r[i+1],search(now));
insert(now);
}
for(int i=1;i<n;i++){
ans=max(ans,l[i]+r[i+1]);
}
printf("%d\n",ans);
return 0;
}
[学习笔记]我们追过的神奇异或(Trie树系列)的更多相关文章
- VC++ 学习笔记(序):神一样的语言
总的来说,我觉得VC++是一门神一样的语言——它是公认最强大.最复杂的:它一切以效率为第一要务,却又不肯落伍,拼命兼容现在的新的语言设计特点.本来在别的语言很容与就避开的问题,在这里要用很高的技巧去设 ...
- Angular 学习笔记 ( 我追的 feature 和 bug )
Angular 有许多功能还不齐全,也有不少 bug 让人很头疼,所以这里做一些记入 Angular Bug 1.input type="number", valueChanges ...
- Unity3D之Mecanim动画系统学习笔记(九):Blend Tree(混合树)
认识Blend Tree 我们在Animator Controller中除了可以创建一个State外还可以创建一个Blend Tree,如下: 那么我们看下新创建的Blend Tree和State有什 ...
- MongoDB学习笔记~为IMongoRepository接口添加了排序和表达式树,针对官方驱动
回到目录 MongoDB的官方驱动,语法更好,更强 之前写过关于IMongoRepository仓储的文章,在mongodb的驱动选择上使用了NoRM,但是进行$ref引用类型导航时出现了问题,它对引 ...
- [学习笔记]dsu on a tree(如何远离线段树合并)
https://www.zybuluo.com/ysner/note/1318613 背景 这玩意来源于一种有局限性的算法. 有一种广为人知的,树上离线维护子树信息的做法. (可以参照luogu360 ...
- 【数据结构与算法Python版学习笔记】递归(Recursion)——定义及应用:分形树、谢尔宾斯基三角、汉诺塔、迷宫
定义 递归是一种解决问题的方法,它把一个问题分解为越来越小的子问题,直到问题的规模小到可以被很简单直接解决. 通常为了达到分解问题的效果,递归过程中要引入一个调用自身的函数. 举例 数列求和 def ...
- MongoDB学习笔记系列
回到占占推荐博客索引 该来的总会来的,Ef,Redis,MVC甚至Sqlserver都有了自己的系列,MongoDB没有理由不去整理一下,这个系列都是平时在项目开发时总结出来的,希望可以为各位一些帮助 ...
- MongoDB学习笔记系列~目录
MongoDB学习笔记~环境搭建 (2015-03-30 10:34) MongoDB学习笔记~MongoDBRepository仓储的实现 (2015-04-08 12:00) MongoDB学习笔 ...
- LM3S之boot loader学习笔记-1
LM3S之boot loader学习笔记-1 彭会锋 (首先声明,此系列文章编写参考了很多资料,其中一些内容是原版内容的引用和整理,并加入了一些自己的见解,我已经尽量标明引用部分,如有未全部标注部分, ...
随机推荐
- C# 创建精简版IIS
1. 方法 一 using System; using System.Collections.Generic; using System.Text; using System.Threading; u ...
- HTML and CSS学习概述
一·Web浏览器是一个连接到Web服务器,向Web服务器请求信息,然后解析返回来的HTML标记,并将其显示在浏览器窗口内的程序.1.Microsoft 2.Internet Explorer(IE)3 ...
- 数组方法indexOf & lastIndexOf
indexOf() 语法:arrayObject.indexOf(searchvalue, startIndex) 功能:从数组的开头(位置0)开始向后查找. 参数:searchvalue:必需,要查 ...
- zabbix实现企业微信监控报警
一.zabbix基本说明 简介:zabbix基于Web界面的分布式系统监控的企业级开源软件.可以监控各种系统与设备,网络参数,保证服务器设备安全运营:提供灵活的通知机制.如果检测到的指标不达标,就实现 ...
- KBMMW 4.84.00 发布
kbmMW is a portable, highly scalable, high end application server and enterprise architecture integr ...
- python使用Fabric模块实现自动化运维
简介:Fabric是基于Python实现的SSH命令行工具,简化了SSH的应用程序部署及系统管理任务,它提供了系统基础的操作组件,可以实现本地或远程shell命令,包括:命令执行.文件上传.下载及完整 ...
- JS页面跳转大全
所谓的js页面跳转就是利用javesrcipt对打开的页面ULR进行跳转,如我们打开的是A页面,通过javsrcipt脚本就会跳转到B页面.目前很多垃圾站经常用js跳转将正常页面跳转到广告页面,当然也 ...
- codeforces题目合集(持续更新中)
CF280CCF280CCF280C 期望dp CF364DCF364DCF364D 随机化算法 CF438DCF438DCF438D 线段树 CF948CCF948CCF948C 堆 CF961EC ...
- 2018.11.08 UVA11021 Tribles(概率dp)
传送门 概率dpdpdp简单题. 设f[i]f[i]f[i]表示第iii天的答案. 然后枚举ppp数组从fi−1f_{i-1}fi−1转移过来就行了. 显然有fi=∑j=0npj∗(fi−1)jf_ ...
- 2018.11.01 loj#2319. 「NOIP2017」列队(线段树)
传送门 唉突然回忆起去年去noipnoipnoip提高组试水然后省二滚粗的悲惨经历... 往事不堪回首. 所以说考场上真的有debuffdebuffdebuff啊!!!虽然当时我也不会权值线段树 这道 ...