[十二省联考2019] 异或粽子 解题报告 (可持久化Trie+堆)
interlinkage:
https://www.luogu.org/problemnew/show/P5283
description:
solution:
- 显然有$O(n^2)$的做法,前缀和优化一下即可
- 正解做法是先确定一个右端点$r$,找到最优的$l$使得该区间的异或和最大,这个可以用可持久化$Trie$实现。不懂的话可以在我的博客里搜索
- 对每个点取出来后把答案放进一个堆里,显然当前的堆顶一定会对答案产生贡献
- 然后我们考虑每次取出的右端点,它依旧可能产生贡献。即上一次取的最优的$l$把原来的区间割裂成了两部分,把这两部分的最优解算出来再次放进堆里就好
- 这样取出$k$个就一定是最优的$k$个了
code:
#include<iostream>
#include<cstring>
#include<cstdio>
#include<queue>
#include<cmath>
#include<algorithm>
using namespace std;
typedef long long ll; const int N=5e5+;
int n,k,cnt;
int rt[N];
ll a[N];
struct Trie
{
int ch[];
int id,siz;
}t[N*];
inline ll read()
{
char ch=getchar();ll s=,f=;
while (ch<''||ch>'') {if (ch=='-') f=-;ch=getchar();}
while (ch>=''&&ch<='') {s=(s<<)+(s<<)+ch-'';ch=getchar();}
return s*f;
}
void ins(int &now,int pre,int bit,int id,ll val)
{
now=++cnt;t[now]=t[pre];t[now].siz++;
if (bit==-)
{
t[now].id=id;
return;
}
if ((val>>bit)&) ins(t[now].ch[],t[pre].ch[],bit-,id,val);
else ins(t[now].ch[],t[pre].ch[],bit-,id,val);
}
int query(int k1,int k2,int bit,ll val)
{
if (bit==-) return t[k2].id;
int d=(val>>bit)&;
if (t[t[k2].ch[d^]].siz-t[t[k1].ch[d^]].siz>) return query(t[k1].ch[d^],t[k2].ch[d^],bit-,val);
return query(t[k1].ch[d],t[k2].ch[d],bit-,val);
}
struct node
{
int l,r,x,id;ll val;
node(int _l=,int _r=,int _x=)
{
l=_l;r=_r;x=_x;
id=query(rt[l-],rt[r],,a[x]);
val=a[x]^a[id-];
}
};
bool operator < (const node &a,const node &b) {return a.val<b.val;}
priority_queue<node> Q;
int main()
{
freopen("xor.in","r",stdin);
freopen("xor.out","w",stdout);
n=read();k=read();
for (int i=;i<=n;i++) a[i]=a[i-]^read();
for (int i=;i<=n;i++) ins(rt[i],rt[i-],,i,a[i-]);
for (int i=;i<=n;i++) Q.push(node(,i,i));
ll ans=;
while (k--)
{
node u=Q.top();Q.pop();
ans+=u.val;
if (u.l<u.id) Q.push(node(u.l,u.id-,u.x));
if (u.r>u.id) Q.push(node(u.id+,u.r,u.x));
}
printf("%lld\n",ans);
return ;
}
[十二省联考2019] 异或粽子 解题报告 (可持久化Trie+堆)的更多相关文章
- [十二省联考2019]异或粽子——可持久化trie树+堆
题目链接: [十二省联考2019]异或粽子 求前$k$大异或区间,可以发现$k$比较小,我们考虑找出每个区间. 为了快速得到一个区间的异或和,将原序列做前缀异或和. 对于每个点作为右端点时,我们维护出 ...
- 【BZOJ5495】[十二省联考2019]异或粽子(主席树,贪心)
[BZOJ5495][十二省联考2019]异或粽子(主席树,贪心) 题面 BZOJ 洛谷 题解 这不是送分题吗... 转异或前缀和,构建可持久化\(Trie\). 然后拿一个堆维护每次的最大值,每次如 ...
- [十二省联考2019]异或粽子 01trie
[十二省联考2019]异或粽子 01trie 链接 luogu 思路 首先求前k大的(xo[i]^xo[j])(i<j). 考场上只想到01trie,不怎么会写可持久,就写了n个01trie,和 ...
- 【简】题解 P5283 [十二省联考2019]异或粽子
传送门:P5283 [十二省联考2019]异或粽子 题目大意: 给一个长度为n的数列,找到异或和为前k大的区间,并求出这些区间的异或和的代数和. QWQ: 考试时想到了前缀异或 想到了对每个数按二进制 ...
- [十二省联考2019]异或粽子(堆+可持久化Trie)
前置芝士:可持久化Trie & 堆 类似于超级钢琴,我们用堆维护一个四元组\((st, l, r, pos)\)表示以\(st\)为起点,终点在\([l, r]\)内,里面的最大值的位置为\( ...
- 洛谷.5283.[十二省联考2019]异或粽子(可持久化Trie 堆)
LOJ 洛谷 考场上都拍上了,8:50才发现我读错了题=-= 两天都读错题...醉惹... \(Solution1\) 先求一遍前缀异或和. 假设左端点是\(i\),那么我们要在\([i,n]\)中找 ...
- 【洛谷5283】[十二省联考2019] 异或粽子(可持久化Trie树+堆)
点此看题面 大致题意: 求前\(k\)大的区间异或和之和. 可持久化\(Trie\)树 之前做过一些可持久化\(Trie\)树题,结果说到底还是主席树. 终于,碰到一道真·可持久化\(Trie\)树的 ...
- Luogu P5283 / LOJ3048 【[十二省联考2019]异或粽子】
联考Day1T1...一个考场上蠢了只想到\(O(n^2)\)复杂度的数据结构题 题目大意: 求前\(k\)大区间异或和的和 题目思路: 真的就是个sb数据结构题,可持久化01Trie能过(开O2). ...
- Luogu5283 十二省联考2019异或粽子(trie/可持久化trie+堆)
做前缀异或和,用堆维护一个五元组(x,l,r,p,v),x为区间右端点的值,l~r为区间左端点的范围,p为x在l~r中最大异或和的位置,v为该最大异或和,每次从堆中取出v最大的元素,以p为界将其切成两 ...
随机推荐
- Aspose.Words进行Word替换(插入图片和水印)
由于最近一直在忙着做着Word打印模板的一些工作,就整理一些Asponse.Words对Word文档进行操作的资料. using System; using System.Collections.Ge ...
- 6.Renderer Window
渲染是实时的,所见即所得.同时还可以输出一些统计信息. Pixel Snoop:获取颜色值,同时把该值复制到剪贴板,主要用户是获取颜色值 Wireframe:开启后可以查看3D节点图形骨架 Stati ...
- 【JavaScript从入门到放弃】JS基础-01
作为一个前端开发人员,JS是我们行走江湖吃饭的家伙.基本上一个前端人员能值多少大洋,就看JS了.虽然各种框架层出不穷,但是归根结底学好原生JS才是硬道理. 学习任何新东西其实都遵循 10000 小时成 ...
- dotnetnuke7.3.3 下弹出对话框(dnnConfirm())的使用
今天用dnn做一个列表里边有一个删除操作,就想做个对话框确定是否删除? 正常理解马上想到js的confirm("")函数,但是发现Dnn把这个函数给重写啦,弹出的对话框竟然是英文的 ...
- JVM内存划分以及值传递和引用传递的区别
Day05_SHJavaTraing_4-8-2017 一.JVM对自己的内存划分为5个区域 1.方法栈:所有的方法运行的时候进入内存 2.堆:存储的是容器和对象 3.方法和数据共享 ...
- Postfix Self Expression
Postfix Self Expression A postfix self expression consists of an expression or the name of a type, i ...
- 有关详细信息, 请使用 -Xlint:unchecked 重新编译。
这是在复制代码的时候,没有修改路径,但是IDEA没有报错,还会爆出 WARN ework.web.servlet.PageNotFound - No mapping found for HTTP re ...
- LINUX KERNEL启动参数
LINUX KERNEL启动参数 在Linux中,给kernel传递参数以控制其行为总共有三种方法: 1.build kernel之时的各个configuration选项. 2.当kernel启动之时 ...
- 更新时间戳.txt
UPDATE bbs2 INNER JOIN time1 ON bbs2.AnnounceID = time1.AnnounceID SET bbs2.asptime = time1.asptime
- Vue学习之路第十四篇:v-for指令中key的使用注意事项
1.学前准备: JavaScript中有一个方法:unshift() ,其作用是向数组的开头添加一个或更多元素,并返回新的长度.该方法的第一个参数将成为数组的新元素 0,如果还有第二个参数,它将成为新 ...