P7470-[NOI Online 2021 提高组]岛屿探险【Trie,CDQ分治】
正题
题目链接:https://www.luogu.com.cn/problem/P7470
题目大意
给出\(n\)个二元组\((a,b)\)。
\(q\)次询问给出\((l,r,c,d)\)表示询问\([l,r]\)中有多少二元组满足\(c\ xor\ a\leq min(b,d)\)。
\(1\leq n,q\leq 10^5\)
解题思路
这个\(min\)一看就很迷,显然是让我们分两种情况讨论。
再把询问拆一下,就变成了两个条件\(pos\leq r/pos<l\)且\(b\leq d/b>d\)。
两个偏序条件的话直接上\(CDQ\),然后考虑两种情况怎么处理。
- \(c\ xor\ a\leq b\):这样对于每个二元组合法的\(c\)开业被拆成\(Trie\)上最多\(log\)个区间,建\(Trie\)即可
- \(c\ xor\ a\leq d\):对于每组询问在\(Trie\)上跑区间求和即可。
时间复杂度\(O(n\log^2 n)\)
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
const int N=1e5+10,M=N*24;
struct node{
int w,l,id;
}q[N<<1],a[N];
int n,m,tot,rt1,rt2,ans[N];
vector<node> v[N];
struct Trie1{
int cnt,ch[M][2],w[M];
void Clear(){rt1=0;cnt=0;return;}
int Newp(){++cnt;ch[cnt][0]=ch[cnt][1]=w[cnt]=0;return cnt;}
void Insert(int &x,int d,int l,int val){
if(!x)x=Newp();
if(d<0){w[x]++;return;}
int c=(val>>d)&1;
if((l>>d)&1){
Insert(ch[x][c^1],d-1,l,val);
if(!ch[x][c])ch[x][c]=Newp();
w[ch[x][c]]++;
}
else Insert(ch[x][c],d-1,l,val);
}
int Ask(int x,int d,int val){
if(!x)return 0;
if(d<0)
return w[x];
int c=(val>>d)&1;
return Ask(ch[x][c],d-1,val)+w[x];
}
}T1;
struct Trie2{
int cnt,ch[M][2],w[M];
void Clear(){rt2=0;cnt=0;return;}
int Newp(){++cnt;ch[cnt][0]=ch[cnt][1]=w[cnt]=0;return cnt;}
void Insert(int &x,int d,int val){
if(!x)x=Newp();
if(d<0){w[x]++;return;}
int c=(val>>d)&1;
Insert(ch[x][c],d-1,val);
w[x]=w[ch[x][0]]+w[ch[x][1]];
return;
}
int Ask(int x,int d,int l,int val){
if(d<0)return w[x];
int c=(val>>d)&1;
if((l>>d)&1)
return Ask(ch[x][c^1],d-1,l,val)+w[ch[x][c]];
return Ask(ch[x][c],d-1,l,val);
}
}T2;
bool cmp(node x,node y)
{return x.l<y.l;}
void CDQ(int l,int r){
if(l==r)return;
int mid=(l+r)>>1;
CDQ(l,mid);
CDQ(mid+1,r);
sort(a+l,a+mid+1,cmp);
T1.Clear();T2.Clear();tot=0;
for(int i=mid+1;i<=r;i++)
for(int j=0;j<v[i].size();j++)
q[++tot]=v[i][j];
sort(q+1,q+1+tot,cmp);
for(int i=1,z=l;i<=tot;i++){
while(z<=mid&&a[z].l<=q[i].l)
T1.Insert(rt1,23,a[z].l,a[z].w),z++;
if(q[i].id<0)ans[-q[i].id]-=T1.Ask(rt1,23,q[i].w);
else ans[q[i].id]+=T1.Ask(rt1,23,q[i].w);
}
for(int i=tot,z=mid;i>=1;i--){
while(z>=l&&a[z].l>q[i].l)
T2.Insert(rt2,23,a[z].w),z--;
if(q[i].id<0)ans[-q[i].id]-=T2.Ask(rt2,23,q[i].l,q[i].w);
else ans[q[i].id]+=T2.Ask(rt2,23,q[i].l,q[i].w);
}
return;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
scanf("%d%d",&a[i].w,&a[i].l);
for(int i=1;i<=m;i++){
int l,r,c,d;
scanf("%d%d%d%d",&l,&r,&c,&d);
v[l].push_back((node){c,d,-i});
v[r+1].push_back((node){c,d,i});
}
sort(q+1,q+1+n,cmp);
CDQ(1,n+1);
for(int i=1;i<=m;i++)
printf("%d\n",ans[i]);
return 0;
}
P7470-[NOI Online 2021 提高组]岛屿探险【Trie,CDQ分治】的更多相关文章
- [NOI Online 2021 提高组] 积木小赛
思路不说了. 想起来自己打比赛的时候,没睡好.随便写了个\(HASH\),模数开小一半分都没有. 然后学了\(SAM\),发现这个判重不就是个水题. \(SAM\)是字串tire的集合体. 随便\(d ...
- CCF NOI Online 2021 提高组 T3 岛屿探险(CDQ 分治,Trie 树)
题面 凇睦是一个喜欢探险的女孩子,这天她到一片海域上来探险了. 在这片海域上一共有 n 座岛屿排成一排,标号为 1, 2, 3, . . . , n.每座岛屿有两个权值,分别为劳累度 ai 和有趣度 ...
- CCF NOI Online 2021 提高组 赛后心得
T1 做个,不会,拿到 20 pts 跑路. 注意后面有个 K = 1 的部分分,这个可以递推求 b 的个数,然后直接乘上 a0 . 官方正解讲得极其详细,我还是第一次见到可以 O(K2) 做 1~n ...
- CCF NOI Online 2021 提高组 T2 积木小赛 (子序列自动机+后缀自动机,O(n^2))
题面 Alice 和 Bob 最近热衷于玩一个游戏--积木小赛. Alice 和 Bob 初始时各有 n 块积木从左至右排成一排,每块积木都被标上了一个英文小写字母. Alice 可以从自己的积木中丢 ...
- luogu P6570 [NOI Online #3 提高组]优秀子序列 二进制 dp
LINK:P6570 [NOI Online #3 提高组]优秀子序列 Online 2的T3 容易很多 不过出于某种原因(时间不太够 浪了 导致我连暴力的正解都没写. 容易想到 f[i][j]表示前 ...
- P7473 [NOI Online 2021 入门组] 重力球
P7473 [NOI Online 2021 入门组] 重力球 题意 给你一个正方形平面,某些位置有障碍,对于平面上两个球,每次你可以改变重力方向使两个球下落到最底端,求使两个球位置重合的最小改变重力 ...
- [NOI Online #2 提高组]涂色游戏 题解
题目描述 你有 1020 个格子,它们从 0 开始编号,初始时所有格子都还未染色,现在你按如下规则对它们染色: 编号是 p1 倍数的格子(包括 0号格子,下同)染成红色. 编号是 p2 倍数的格子染成 ...
- NOI Online #2 提高组 游戏
没用二项式反演的菜比. 题目链接 Solution 非平局代表的树上祖先关系是比较好统计,(可以在处理一个点时,考虑用他去匹配他的子树中的东西)而平局的关系比较难统计.我们不妨求出至少 \(k\) 个 ...
- NOI Online #2 提高组 游记
没 NOI Online 1 挂的惨就来写游记吧,不知道为啥 NOI Online 1 民间数据测得 60 分的 T1 最后爆零了... 昏昏沉沉的醒来,吃了早饭,等到 \(8:30\) 进入比赛网页 ...
随机推荐
- join控制线程的执行循序 T1 -> T2 -> T3
/** * 控制线程的执行循序 T1 -> T2 -> T3 * join实现 */ public static void join(){ Thread t1 = new Thread(( ...
- SpringBoot枚举传参
创建一个接口所有枚举继承 package com.gecko.charging.common; public interface BaseEnum { Integer getCode(); } 具体的 ...
- @ControllerAdvice注解(全局异常捕获)
背景 @ControllerAdvice 注解 通常用于定义@ExceptionHandler, @InitBinder和@ModelAttribute 适用于所有@RequestMapping方法的 ...
- html,javascript,正则表达式
正则表达式是对字符串操作的逻辑公式,用事先定好的一些特定字符组成一个"规则字符串",在用"规则字符串"对字符串进行过滤. ECMAScript 通过RegExp ...
- Socket通信协议解析(文章摘要)
参考网址: https://zhuanlan.zhihu.com/p/84800923 在计算机通信领域,socket 被翻译为"套接字",它是计算机之间进行通信的一种约定或一种方 ...
- C# 不是异步的方法中获取异步的结果
var waiter = HP.UtilsLib.TaskAwaiterHelper.GetTaskAwaiter( async () => { return await feedBack(ve ...
- C#设计模式---PipeLine
一.概述 顾名思义,管道模式就像一条管道把多个对象连接起来,整体看起来就像若干个阀门嵌套在管道中,而处理逻辑就放在阀门上,如下图,需要处理的对象进入管道后,分别经过阀门一.阀门二.阀门三.阀门四,每个 ...
- WPF---数据绑定(二)
一.绑定到非UI元素 上篇中,我们绑定的数据源均是派生自UIElement的WPF元素.本篇描述的绑定数据源是一个我们自定义的普通的类型. 注:尽管绑定的数据源可以是任意类型的对象,但Path必须总是 ...
- (3)hadoop单节点配置
hadopp版本hadoop-2.6.0-cdh5.7.0 下载地址http://archive-primary.cloudera.com/cdh5/cdh/5/ 同样使用flashFXP上 ...
- JavaScript高级程序设计读书笔记之JSON
JSON(JavaScript Object Notation)JavaScript对象表示法.JSON是JavaScript的一个严格的子集,利用了JavaScript中的一些模式来表示结构化数据. ...