Vitya and Strange Lesson

Vitya and Strange Lesson

https://ac.nowcoder.com/acm/problem/112209

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());
    }
}
有的没的 文章被收录于专栏

RT,有的没的

全部评论
大佬太强了!
1 回复 分享
发布于 2020-10-25 09:16
大佬太强了!
点赞 回复 分享
发布于 2020-10-25 10:20

相关推荐

已oc&nbsp;云智断更了好几天,也有一些话想说,继续更新一篇云智timeline&nbsp;4.18&nbsp;一面&nbsp;半个小时后约二面&nbsp;4.21二面&nbsp;当晚&nbsp;约hr面&nbsp;4.23hr面&nbsp;4.30&nbsp;发offer之前美团的二面挂了,进入人才库,后面又被捞起来面试,4.30号&nbsp;美团又一面,现在还没出一面结果感觉也不报什么希望,就算一面过了,还有二面,我经不起深入拷打,唉,真的,好难五一躺平了五天,吃吃玩玩睡睡~还要担心毕业,科研更是难,唉暑期可能就到此为止了,后面没有时间在这个上面了,要抓紧时间做科研,为了后面能出去实习。大厂,秋招再见!!!有一些感慨:4.1是我的第一次面试,美团,面试的时候紧张到浑身发...
daisy9542:我今晚也是美团一面,已经第六次了。我也面了其他的,没拿到 offer。但我想开了,要按照自己的节奏来,找暑期转正然后秋招大杀四方并不是唯一的出路,其实还有很多选择的,有 0 实习最后秋招拿 offer 了,也有不选择互联网去国企的外企的,考编的,创业的。现在的失败不代表以后的路都是黑暗的,只不过可能运气还没降临到头上。所以现在要做的,就是放平心态,提升自己,通过面试了解到自己的优点和不足,争取下次机会来了能好好抓住
点赞 评论 收藏
分享
评论
4
3
分享

创作者周榜

更多
牛客网
牛客企业服务