BZOJ 4103 [Thusc 2015]异或运算 (可持久化01Trie+二分)
题目大意:给你一个长方形矩阵,位置$i,j$上的数是$a_{i}\;xor\;b_{j}$,求某个子矩阵内第$K$大的值
最先想的是二分答案然后验证,然而是$O(qnlogmloga_{i})$,不出意外会被卡..看完题解才恍然大悟
$01Trie$是具有二分性质的!因为每个节点最多有2个儿子!
先对$b$序列建可持久化$01Trie$,记录一个$sum$表示当前节点的子树内有多少个数
对于每次询问,因为$n$很小,暴力枚举$a$进行统计,记录每个a当前在01Trie的位置
接下来就是在$01Trie$上二分了
按位从高到低枚举,统计一共有多少个数这一位是1,即每个a所在$01Trie$位置 和这一位异或值为1 的子树内,记为$tot$
如果$tot>k$,说明这一位不是1,每个$a$分别向异或值是0的地方走,然后$K-=tot$,去掉这一位填1的贡献
反之每个$a$往异或值为1的地方走
最后输出答案即可
#include <set>
#include <queue>
#include <vector>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N1 301000
#define N2 10201000
#define MM 100
#define ll long long
#define dd double
#define uint unsigned int
#define mod 1000000007
#define idx(X) (X-'a')
#define it multiset<node>::iterator
using namespace std; int gint()
{
int ret=,fh=;char c=getchar();
while(c<''||c>''){if(c=='-')fh=-;c=getchar();}
while(c>=''&&c<=''){ret=ret*+c-'';c=getchar();}
return ret*fh;
} uint bin[];
int n,m;
uint a[N1],b[N1],pa[N1];
int px[N1],py[N1]; struct Trie{
int ch[N2][],num[N2],root[N1],tot;
int sum[N2],stk[N2],tp;
void init()
{
root[]=tot=;int x=;
for(int i=;i>=;i--){
ch[x][]=++tot;
x=ch[x][],num[x]=;
}
}
void insert(int s,int rt1,int rt2,int w)
{
int x,y,p;
y=root[rt1];
x=root[rt2]=++tot;
for(int i=;i>=;i--){
p=(s&bin[i])?:;
ch[x][p]=++tot;
ch[x][p^]=ch[y][p^];
num[ch[x][p]]=num[ch[y][p]]+w;
sum[ch[x][p]]=sum[ch[y][p]];
x=ch[x][p],y=ch[y][p];
stk[++tp]=x;
}
sum[x]++,stk[tp--]=;
while(tp){
x=stk[tp--];
sum[x]=sum[ch[x][]]+sum[ch[x][]];
}
}
uint query(int L,int R,int l,int r,int K)
{
int x,y,p;uint ans=;
y=l<?:root[l],x=root[r];
int s[],tot;
for(int j=L;j<=R;j++)
px[j]=x,py[j]=y;
for(int i=;i>=;i--)
{
s[]=,s[]=,tot=;
for(int j=L;j<=R;j++){
p=(a[j]&bin[i])?:,s[p]++;
tot+=sum[ch[px[j]][p^]]-sum[ch[py[j]][p^]];
}
if(K>tot){
K-=tot;
for(int j=L;j<=R;j++){
p=(a[j]&bin[i])?:;
if(num[ch[px[j]][p]]-num[ch[py[j]][p]]>)
px[j]=ch[px[j]][p],py[j]=ch[py[j]][p];
else px[j]=py[j]=;
}
}else{
for(int j=L;j<=R;j++){
p=(a[j]&bin[i])?:;
if(num[ch[px[j]][p^]]-num[ch[py[j]][p^]]>)
px[j]=ch[px[j]][p^],py[j]=ch[py[j]][p^];
else px[j]=py[j]=;
}ans|=bin[i];
}
}return ans;
}
}T; int main()
{
//freopen("t1.in","r",stdin);
scanf("%d%d",&n,&m);
for(int i=;i>=;i--)
bin[i]=(<<i);
for(int i=;i<=n;i++)
a[i]=gint();
T.init();
for(int i=;i<=m;i++)
b[i]=gint(),T.insert(b[i],i-,i,);
int Q,u,d,l,r,K;
scanf("%d",&Q);
for(int q=;q<=Q;q++)
{
u=gint(),d=gint(),l=gint(),r=gint(),K=gint();
printf("%d\n",T.query(u,d,l-,r,K));
}
return ;
}
BZOJ 4103 [Thusc 2015]异或运算 (可持久化01Trie+二分)的更多相关文章
- 【BZOJ 4103】 [Thu Summer Camp 2015]异或运算 可持久化01Trie
我们观察数据:树套树 PASS 主席树 PASS 一层一个Trie PASS 再看,异或!我们就把目光暂时定在01Tire然后我们发现,我们可以带着一堆点在01Trie上行走,因为O(n*q* ...
- [BZOJ4103][Thu Summer Camp 2015]异或运算 可持久化Trie树
4103: [Thu Summer Camp 2015]异或运算 Time Limit: 20 Sec Memory Limit: 512 MB Description 给定长度为n的数列X={x1 ...
- BZOJ 4103: [Thu Summer Camp 2015]异或运算 可持久化trie
开始想了一个二分+可持久化trie验证,比正解多一个 log 仔细思考,你发现你可以直接按位枚举,然后在可持久化 trie 上二分就好了. code: #include <bits/stdc++ ...
- 【bzoj4103】[Thu Summer Camp 2015]异或运算 可持久化trie树
Description 给定长度为n的数列X={x1,x2,...,xn}和长度为m的数列Y={y1,y2,...,ym},令矩阵A中第i行第j列的值Aij=xi xor yj,每次询问给定矩形区域i ...
- 【BZOJ 4103】 4103: [Thu Summer Camp 2015]异或运算 (可持久化Trie)
4103: [Thu Summer Camp 2015]异或运算 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 474 Solved: 258 De ...
- [BZOJ 4103] [Thu Summer Camp 2015] 异或运算 【可持久化Trie】
题目链接:BZOJ - 4103 题目分析 THUSC滚粗之后一直没有写这道题,从来没写过可持久化Trie,发现其实和可持久化线段树都是一样的.嗯,有些东西就是明白得太晚. 首先Orz ZYF-ZYF ...
- BZOJ 3689 异或之 (可持久化01Trie+堆)
题目大意:给你一个序列,求出第$K$大的两两异或值 先建出来可持久化$01Trie$ 用一个$set$/堆存结构体,存某个异或对$<i,j>$的第二关键字$j$,以及$ai\;xor\;a ...
- BZOJ 3261 最大异或和 (可持久化01Trie)
题目大意:让你维护一个序列,支持在序列末插入一个数,支持询问$[l,r]$区间内选择一个位置$p$,使$xor\sum_{i=p}^{n}a_{i}$最大 可持久化$01Trie$裸题,把 区间异或和 ...
- P5283 [十二省联考2019]异或粽子 可持久化01Trie+线段树
$ \color{#0066ff}{ 题目描述 }$ 小粽是一个喜欢吃粽子的好孩子.今天她在家里自己做起了粽子. 小粽面前有 \(n\) 种互不相同的粽子馅儿,小粽将它们摆放为了一排,并从左至右编号为 ...
随机推荐
- css——权重叠加
权重叠加 在下面的一段代码中,第一个样式body b有两个标签,第二个有一个标签b.两个中都有color,会应用哪一个呢?果是 那下面的代码会显示什么样的结果 结果是 应用的事body b中的colo ...
- POJ 3517 And Then There Was One( 约瑟夫环模板 )
链接:传送门 题意:典型约瑟夫环问题 约瑟夫环模板题:n个人( 编号 1-n )在一个圆上,先去掉第m个人,然后从m+1开始报1,报到k的人退出,剩下的人继续从1开始报数,求最后剩的人编号 /**** ...
- Linux网络编程(一):一个简单的socket程序
服务器: /* *tcp_server.c */ #include <stdio.h> #include <sys/socket.h> #include <netinet ...
- Python及相应软件安装
Python安装 这是下载地址:Linux下载链接,windows下载链接 1.下载压缩包 wget https://www.python.org/ftp/python/3.7.1/Python-3. ...
- Vue中如何在组件内部实现一个双向数据绑定?
假设有一个输入框组件,用户输入时,同步父组件页面中的数据. 具体思路:父组件通过props传值给子组件,子组件通过 $emit 来通知父组件修改相应的props值,具体实现如下: import Vue ...
- css表格表单和统筹
css:表格表单和统筹 学习目标 1.表单标签及属性高级 2.表格标签及属性高级 3.CSS统筹 4.BFC概念和应用场景 一.表单标签及属性高级 回顾: 表单的作用:用来收集用户的信息的; 表单的组 ...
- 数人云CTO解读Docker 1.12和金融业容器化
7月29日 数人云 在上海举办金融沙龙,邀请上交所和近二十家来自银行.保险.证券的IT技术专家一同探讨容器技术在金融业中的最佳实践.数人云CTO肖德时在会上将传统金融行业通过容器可以解决的四大问题做了 ...
- PatentTips - Sleep state mechanism for virtual multithreading
BACKGROUND The present disclosure relates generally to information processing systems and, more spec ...
- centos7修改网卡名
http://blog.csdn.net/henulwj/article/details/47061023
- Swift开发教程--怎样使UITableViewController背景透明
self.tableView.backgroundView? .backgroundColor = UIColor.clearColor(); self.tableView.backgroundCol ...