AcWing 256. 最大异或和

AcWing 256. 最大异或和

一、题目大意

给定一个非负整数序列 a,初始长度为 N

M 个操作,有以下两种操作类型:

A x:添加操作,表示在序列末尾添加一个数 x,序列的长度 N 增大 1
Q l r x:询问操作,你需要找到一个位置 p,满足 lpr,使得:a[p]a[p+1]a[N]x 最大,输出这个最大值

输入格式
第一行包含两个整数 NM,含义如问题描述所示。

第二行包含 N非负整数,表示初始的序列 A

接下来 M 行,每行描述一个操作,格式如题面所述。

输出格式
每个询问操作输出一个整数,表示询问的答案。

每个答案占一行。

数据范围
N,M3×105,0a[i]107

输入样例:

5 5
2 6 4 3 6
A 1 
Q 3 5 4 
A 4 
Q 5 7 0 
Q 3 6 6 

输出样例:

4
5
6

二、前导知识

异或问题

异或问题是研究数列上异或性质的一类问题,例如 区间最大异或异或和 相关问题等,解决这些问题通常用到下面的几个性质:

  • 交换律 ab=ba
  • 结合律 (ab)c=a(bc)
  • 自反性 xx=0
  • 或 0 不变性 x0=x

根据自反性质,区间的异或值具有前缀和性质,即

i=lrai=(i=1l1ai)(i=1rai)

因此我们可以更方便地处理问题。

证明:
S(x)=a1a2...ax

S(r)=a1a2...arS(l1)=a1a2...al1S(r)S(l1)=alal+1...ar

可持久化Trie

Q:本题涉及到的是异或运算和,使用Tire树是可以理解的,但为什么一定要持久化,不持久化为什么不行?

A: 之所以选择可持久化Trie来完成这道题,原因是:

  • 普通最大异或值可以通过构建普通Trie,一路能反着走就反着走,实在走不了就正着走,来获取,这是一个贪心的思想
  • 普通Trie无法解决区间[L,R]这样的查询问题,一查就是全套的,不知道什么进候收手
  • 如果记录并枚举从L~R的每一个Trie树,就在空间和时间上过不去,这时,持久化Trie树登场
  • [1R1]可以直接查找版本号为R1的数据,不会取到大于等于R1的数据
  • [L1R1]的数据,其实在版本为R1的树中其实都存在的,但直接取怕到[1L2]中去,造成错误查询, 办法就是在每个节点创建时,标识它是由哪个版本创建的,如果是>=L1的,才能访问

可持久化:下面介绍 Trie 是如何实现可持久化的。

既然我们现在要访问一个历史版本,那么我们直观的想法就是将每一个版本的 Trie 结构体都存储下来,当需要一个新的结点时,我们完全复制一个历史版本,然后再它上面完成操作。这样的做法,正确性是显然的,但是空间开销却让人头疼。当务之急是减少存储空间,我们考虑将 Trie 树上的一些 枝条 共用来减少空间上的浪费。

这样做:对于一个新建的版本,每插入一个点都新建一个节点,然后完全复制历史版本上同等地位点的全部儿子信息,可以看下面这张图来方便你的理解。

通过上图我们发现,从一个版本起点开始,遍历整棵树,一定只能获得该版本内的所有串,并且空间大大减少,是不是非常优美。

对于区间 [l,r]上的一些询问,我们转化为对版本l1r之间插值的询问。这样就可以通过可持久化的方法来求解区间信息。

图集

1. Trie树中保存的是什么?
2. 可持久化Trie的构建步骤

四、本题思路

定义Si表示前i个数的 异或前缀和,即:

S0=0 S1=a1 S2=a1a2  Si=a1a2a3......ai

需要求解的内容变为:

ap......anx=Sp1Snx

上面的式子中可以将Snx看成常数,记为C,则相当于在区间[L,R]中找到一个位置p,使得Sp1C的值 最大

类似于AcWing 143. 最大异或对

将每个数据ai看成一个 二进制字符串,存入到Trie。因为0a[i]107,又223a[i]224,因此我们需要将每个数据对应到一个长度为24为的01二进制串上。

先考虑简单情况
假设让我们从[1,R]中找到一个这样的p的话,问题就十分类似于AcWing 143. 最大异或对不同点 在于本题中的a数组是不断变化的,维护一个Trie树,只能计算某个时刻问题。

因此要记录下所有历史版本的Trieroot[R]中存储的就是插入a[1R]时形成的Trie树。

小结

  • 利用可持久化的Trie树这种数据结构,可以实现从1R区间查询。,其中R也就是版本号,也就是第R个插入的字符串。

  • 如果区间左边的限制也加上,则问题就变成了让我们在区间[L,R]中找到一个p,使得Sp1C的值最大,可以这样处理:在trie树中的每个节点中多记录一个信息ver表示第几个版本插入的,也就是第几个数时插入的,如果ver[u]L,则说明这棵子树在[L,R]这个区间中存在。

  • 对于上面提到的某个C,数据A可以看成一个24位长度的二进制字符串,从左到右遍历这个字符串,假设当前考察的是字符t,则在trie树中我们应该走到t ^ 1的分支上(如果存在的话,即对于区间[L,R],如果该分支对应的ver[u]L,则说明存在),这样异或值才能最大(贪心思想)

  • 原序列长度为3×105,因为操作的个数最多也是3×105,因此序列的长度最大是6×105。另外还需要考虑trie中节点的个数:每次操作最多建立24个节点,再加上根节点,一共25个节点,每个数据最多建立25个节点,因此节点数为25×6×105=1.5×107,每个节点两个分支,因此第二维为2;另外还需要记录每个点的ver,需要的空间量是1.5×107×3=4.5×107int,大约4.5×107×4/106=180MB的存储空间,题目提供256MB的存储空间,满足要求

五、实现代码

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <iostream>
#include <cmath>

using namespace std;

const int N = 6e5 + 10, M = 25 * N;
int s[N];
int tr[M][2], ver[M];
int root[N], idx;

void insert(int k, int p, int q) {
    for (int i = 23; ~i; i--) {
        int u = s[k] >> i & 1;
        tr[q][u ^ 1] = tr[p][u ^ 1]; //复制
        tr[q][u] = ++idx;            //创建
        ver[tr[q][u]] = k;           //记录版本
        q = tr[q][u], p = tr[p][u];
    }
}

int query(int p, int l, int c) {
    for (int i = 23; ~i ; i--) {
        int u = c >> i & 1;
        if (tr[p][u ^ 1] && ver[tr[p][u ^ 1]] >= l)
            p = tr[p][u ^ 1];
        else
            p = tr[p][u];
    }
    return c ^ s[ver[p]]; // p:最终停留在的异或和最大值终点处,ver[p]这是哪个版本放进来的?,s[ver[p]]=S_{p-1}
}

int main() {
    ios::sync_with_stdio(false), cin.tie(0);
    int n, m;
    cin >> n >> m;

    // 0号版本,用于处理类似于S[1]-S[0]这样的递推边界值
    root[0] = ++idx;
    insert(0, 0, root[0]);

    for (int i = 1; i <= n; i++) {
        int x;
        cin >> x;
        root[i] = ++idx;
        s[i] = s[i - 1] ^ x; //原数组不重要,异或前缀和数组才重要
        insert(i, root[i - 1], root[i]);
    }

    while (m--) {
        char op;
        cin >> op;
        if (op == 'A') {
            int x;
            cin >> x;
            n++;

            root[n] = ++idx;
            s[n] = s[n - 1] ^ x;
            insert(n, root[n - 1], root[n]);
        } else {
            int l, r, x;
            cin >> l >> r >> x;
            printf("%d\n", query(root[r - 1], l - 1, s[n] ^ x));
        }
    }
    return 0;
}
posted @   糖豆爸爸  阅读(423)  评论(4编辑  收藏  举报
相关博文:
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
历史上的今天:
2018-04-20 关于git中Pageant开机启动且自动关联秘钥
2016-04-20 ASPOSE的示例下载地址
2013-04-20 intellij idea 12 编码不可映射字符
2013-04-20 MySQL性能优化的21个最佳实践
2013-04-20 MySQL MyISAM与Innodb优化方案比较
Live2D
点击右上角即可分享
微信分享提示