Vitya and Strange Lesson
题目大意
就是给定一串序列,然后有一系列全局异或操作,问你每次操作后,没有出现过的最小的非负整数
分析
考虑如何在 字典树上找到最小没有出现过的数
- 每次插入一个数的时候,对形成的路径上的每一个点权值
- 如果这个点对应的是二进制下第 位(最低位为 ),且这个点的权值为 ,那么这颗子树一定是个满二叉树,即这个点的子树中并不存在没有出现过的非负整数,所以这个点不能出现在我们的查询路径中
- 所以如果 的那棵子树满了就可以直接去到 的子树中,这样能保证至少有一棵树不是满二叉树
所以,第一个不在字典树上的数就是我们要求的的答案
然后的问题就是如何做到全局异或一个值呢?
首先来想一想,如果没有全局异或的操作,我们应该如何查询?
按照上面的方法,我们的查询应该是从零开始找,然后在从高位向低位跳的时候,如果 的那棵子树满了,就跳向 的子树,直到找到第一个不在树上的数
那如果有异或呢,我们可以很轻松的得到异或运算也是存在结合律的,所以有
所以说,对于每一次的全局异或,都是可以异或到 上,而不用对 字典树进行修改操作的
所以这道题就顺利解决了
Code
#include <cstdio> const int maxn = 3e5 + 10; int son[maxn * 20][2], id; int size[maxn * 20][2], temp; bool vis[maxn]; inline int __read() { int x(0), t(1); char o (getchar()); while (o < '0' || o > '9') { if (o == '-') t = -1; o = getchar(); } for (; o >= '0' && o <= '9'; o = getchar()) { x = (x << 1) + (x << 3) + (o ^ 48); } return x * t; } inline void insert() { int p(0), x(__read()); if (vis[x]) return; vis[x] = 1; for (int i = 18; ~i; --i) { int sta = x >> i & 1; if (!son[p][sta]) son[p][sta] = ++id; size[p][sta]++; p = son[p][sta]; } } inline int query() { int p(0), res(0); for (int i = 18; ~i; --i) { int sta = temp >> i & 1; if (size[p][sta] == (1 << i)) p = son[p][sta ^ 1], res |= (1 << i); else p = son[p][sta]; if (!p) return res; } return res; } int main() { int n = __read(), m = __read(); for (int i = 1; i <= n; ++i) insert(); while (m--) { temp ^= __read(); printf ("%d\n", query()); } }