Description

给定一个长度为 \(n\) 的序列 \(A\),有 \(m\) 次操作,每次要么在序列尾部再添加一个数,将序列长度 \(n\) 加一,要么给进行一次查询,给定查询参数 \(l,~r,~x\) 要求在 \([l,~r]\) 内找一个位置 \(p\),要求最大化 \(x~~xor ~~Xor_{i = p}^{n} A_i\)。

Limitation

\(1 \leq n,~m \leq 3 \times 10^5\)

\(0 \leq A_i \leq 10^7\)

Solution

考虑由于带加入,后缀异或和不太好维护,于是考虑维护前缀异或和 \(\{presum\}\),那么每次询问等于找一个点 \(p \in [l - 1, r)\),最大化 \(presum_p~~xor~~presum_n~~xor~~x\)。

由于 \(presum_n~~xor~~x\) 是已知的,于是问题就是查询区间内一个位置最大化该点和某个给定值的异或值。 考虑类似主席树的建树方法建立可持久化0/1Trie,在 Trie 的每个节点记录前缀该节点共有几个为 \(0\) 的孩子和为 \(1\) 的孩子,相减即可维护对应区间每个节点是否有对应孩子。

有几个需要注意的细节,首先是由于查的是前缀-1,所以允许如果 $l =1 $ 则前缀什么都不选,因此 \(presum_0\) 应插入一个 \(0\) 而不是置为空。另外注意通过查询的位置 \(x\) 是 \(p - 1\),对应相减的可持久化0/1Tire所维护的版本号应是 \(l - 2,~r - 1\)。当 \(l = 1\) 时需要特判维护版本号是 \(l - 1, r - 1\),同时如果 \(r = 1\) 则对应查询的版本号为 \(0,~0\),当然什么也查不到,应该特判这种情况。另外由于插入 \(m\) 个数,原始 \(n\) 个数,所以数组需要开 \(n + m\)

Code

卡常的题都是屑题

出卡常题的出题人都是毒瘤出题人

好吧被卡的很难受,做了一点优化。

首先是注意到无论是插入还是查询,在0/1Trie上走的路径都是一条链而不具备分叉结构,所以不需要递归只需要循环即可解决问题。但是貌似意义不大,因为递归了写的也是尾递归,貌似开了 -O2 以后会被自动优化成循环233333所以并不能解决问题

写了个内存池,这个玩意真tm有用……甩动态分配内存两条街……

然后就最慢 \(940s\) 卡着过了

#include <cstdio>
#include <algorithm> const int maxn = 600005;
const int upceil = 25;
const int tupceil = 24;
const int maxt = maxn * upceil; struct Tree {
Tree *ch[2];
int v[2]; Tree() { ch[0] = ch[1] = NULL; v[0] = v[1] = 0; }
};
Tree *rot[maxn], *pool[maxt], qwq[maxt]; int n, m, sum, top;
int MU[maxn]; void build(Tree *u, Tree *pre, const int v, int p);
int query(const Tree *const pre, const Tree *const u, const int v, const int p); int main() {
freopen("1.in", "r", stdin);
qr(n); qr(m);
for (int i = 0; i < maxt; ++i) pool[i] = qwq + i;
build(rot[0] = pool[top++], NULL, 0, tupceil);
for (int i = 1, x; i <= n; ++i) {
x = 0; qr(x);
build(rot[i] = pool[top++], rot[i - 1], sum ^= x, tupceil);
}
for (int a, b, c; m; --m) {
a = 0;
do a = IPT::GetChar(); while ((a != 'A') && (a != 'Q'));
if (a == 'A') {
a = 0; qr(a);
build(rot[n + 1] = pool[top++], rot[n], sum ^= a, tupceil);
++n;
} else {
a = b = c = 0; qr(a); qr(b); qr(c);
if (b == 1) {
qw(sum ^ c, '\n', true);
continue;
}
qw(query(rot[std::max(0, a - 2)], rot[b - 1], sum ^ c, tupceil) ^ sum ^ c, '\n', true);
}
}
return 0;
} void build(Tree *u, Tree *pre, const int v, int p) {
while (~p) {
if (pre) *u = *pre;
int x = (v & (1 << p)) >> p;
++u->v[x];
u = u->ch[x] = pool[top++];
if (pre) pre = pre->ch[x];
--p;
}
} int query(const Tree *pre, const Tree *u, const int v, int p) {
int _ret = 0;
while (~p) {
int x = ((v & (1 << p)) >> p) ^ 1, y = u->v[x] - (pre ? pre->v[x] : 0);
if (!y) x ^= 1;
_ret |= (x << p);
u = u->ch[x];
if (pre) pre = pre->ch[x];
--p;
}
return _ret;
}

【可持久化0/1Trie】【P4735】最大异或和的更多相关文章

  1. 洛谷 P4735 最大异或和 解题报告

    P4735 最大异或和 题目描述 给定一个非负整数序列\(\{a\}\),初始长度为\(N\). 有\(M\)个操作,有以下两种操作类型: A x:添加操作,表示在序列末尾添加一个数\(x\),序列的 ...

  2. Codeforces Round #367 (Div. 2) D. Vasiliy's Multiset (0/1-Trie树)

    Vasiliy's Multiset 题目链接: http://codeforces.com/contest/706/problem/D Description Author has gone out ...

  3. Bzoj3261/洛谷P4735 最大异或和(可持久化Trie)

    题面 Bzoj 洛谷 题解 显然,如果让你查询整个数列的最大异或和,建一颗\(01Trie\),每给定一个\(p\),按照二进制后反方向跳就行了(比如当前二进制位为\(1\),则往\(0\)跳,反之亦 ...

  4. P4735 最大异或和 /【模板】可持久化Trie

    //tire的可持久化 //线段树的可持久化——主席树 //可持久化的前提:本身的拓扑结构在操作时不变 //可以存下来数据结构的所有历史版本 //核心思想:只记录每一个版本与前一个版本不一样的地方 / ...

  5. 可持久化+Trie || BZOJ 3261最大异或和 || Luogu P4735 最大异或和

    题面:最大异或和 代码: #include<cstdio> #include<cstring> #include<iostream> using namespace ...

  6. luogu P4735 最大异或和

    嘟嘟嘟 省选竟然考了一个可持久化trie,就挑着我不会的考. 话说考场上我确实写了一个trie的做法,只不过一直没调出来然后就只剩暴力分了. 现在想想实在是太蠢了,明明对算法没有把握,却头脑一热在这题 ...

  7. 51Nod - 1295:XOR key (可持久化Trie求区间最大异或)

    给出一个长度为N的正整数数组A,再给出Q个查询,每个查询包括3个数,L, R, X (L <= R).求ALL 至 ARR 这R - L + 1个数中,与X 进行异或运算(Xor),得到的最大值 ...

  8. P4735 最大异或和 01 Trie

    题目描述 给定一个非负整数序列 \(\{a\}\),初始长度为\(n\). 有 \(m\) 个操作,有以下两种操作类型: \(A\ x\):添加操作,表示在序列末尾添加一个数 \(x\),序列的长度 ...

  9. 【题解】洛谷P4735最大异或和

    学习了一下可持久化trie的有关姿势~其实还挺好理解的,代码也短小精悍.重点在于查询某个历史版本的trie树上的某条边是否存在,同样我们转化到维护前缀和来实现.同可持久化线段树一样,我们为了节省空间继 ...

随机推荐

  1. Composer 入门后,接下来该看看这篇文章了

    上篇文章,我们了解了 Composer 的包是如何开发的,过程中我们使用了 composer init. composer install. composerrequire. composer con ...

  2. 【FPGA】Verilog实现交通信号灯

    大二数字电路的课程设计中,有一份日常作业使用Xilinx FPGA实现简易交通信号灯,但很可惜当时时间有限,没能最终完成.正好在这一学期选修SOPC设计课程,同样采用了Xilinx FPGA,故打算重 ...

  3. 干货最新版 Spring Boot2.1.5 教程+案例合集

    最近发了一系列的 Spring Boot 教程,但是发的时候没有顺序,有小伙伴反映不知道该从哪篇文章开始看起,刚好最近工作告一个小小段落,松哥就把这些资料按照学习顺序重新整理了一遍,给大家做一个索引, ...

  4. Key ssd_300_vgg/block3_box/L2Normalization/gamma not found in checkpoint的解决方案

    在Tensorflow下使用SSD模型训练自己的数据集时,经过查找很多博客资料,已经成功训练出来了自己的模型,但就是在测试自己模型效果的时候,出现了如下错误. 2019-10-27 14:47:12. ...

  5. git的本质是资源库和版本(资源)目录的维护过程

    仓库的本质: 资源+索引. 对git而言,添加到暂存区的过程是,将单个资源的修改副本保存到资源库,同时维护暂存区目录的过程. git的本质是资源库和版本目录的维护过程. 一.要素 1.资源 2.副本 ...

  6. JavaScript 加减危机——为什么会出现这样的结果?

    在日常工作计算中,我们如履薄冰,但是 JavaScript 总能给我们这样那样的 surprise~ 0.1 + 0.2 = ? 1 - 0.9 = ? 如果小伙伴给出内心的结果: 0.1 + 0.2 ...

  7. Markdown温故知新(2):详解七大标准语法

    1.标题 2.强调(粗体 & 斜体) 3.列表 4.链接 & 图片 5.代码块 6.引用 7.分割线 8.总结 说明:在 Markdown 中,一行是指由换行符或回车符之外的字符构成的 ...

  8. EHLIB 安装方法

    Ehlib安装方法  路人甲 2010-05-05 23:01:37 安装文件自带的Readme.txt中的安装过程如下: 1. Delphi 5.x - 7.x, Delphi 9.X Win32, ...

  9. Vue开发日志

    一 搭建环境 mac 安装node brew install node 安装vue 全家桶 npm install -g vue-cli 选择一个目录做初始化 vue init webpack myp ...

  10. three 3D实例学习

    <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...