蓝桥杯真题 - 异或和之差 - 题解

题目链接:https://www.lanqiao.cn/problems/3524/learning/

个人评价:难度 3 星(满星:5)

前置知识:前缀和,字典树


整体思路

  • 定义 l m a x i lmaxi lmaxi 为所有右端点小于等于 i i i 的区间异或和的最大值, r m a x i rmaxi rmaxi 为所有左端点大于等于 i i i 的区间异或和最大值,同理定义 l m i n i lmini lmini 与 r m i n i rmini rmini,则答案如下:

a n s = max ⁡ ( l m a x i − r m i n i + 1 , r m a x i + 1 − l m i n i ) i ∈ 2 , n ans=\max(lmaxi - rmini + 1, rmaxi + 1 - lmini) \quad i\in2, n ans=max(lmaxi−rmini+1,rmaxi+1−lmini)i∈2,n

  • 考虑 l m a x lmax lmax 数组的维护( l m i n , r m i n , r m a x lmin, rmin, rmax lmin,rmin,rmax 数组同理),在枚举到第 i i i 个数字时,如果能快速找到以 i i i 为右端点的所有区间异或和最大值,再将这个最大值与 l m a x i − 1 lmaxi - 1 lmaxi−1 取 m a x max max,就是 l m a x i lmaxi lmaxi 的值;
  • 定义 s u m i sumi sumi 为前 i i i 个数字的前缀异或和,将 s u m i sumi sumi 与所有 s u m j ( j ∈ 0 , i ) ) sum\[j~(j \in 0, i)) sum\[j (j∈0,i)) 异或和的最大值即第二点所求值,将所有 s u m \[ j sumj sumj 放入 01 01 01 字典树中,即可 O ( log ⁡ n ) O(\log n) O(logn) 时间复杂度求出该最大值。

过题代码

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

typedef long long LL;
const int maxn = 200000 + 100;
const int maxcnt = 200000 * 21 + 100;
const int Size = 2;

// 01 Trie 模板代码
struct Trie {
    int root, cnt;
    int tree[maxcnt][Size];

    int create() {
        ++cnt;
        memset(tree[cnt], 0, sizeof(tree[cnt]));
        return cnt;
    }

    void Init() {
        cnt = 0;
        root = create();
    }

    void add(int x) {
        int pos = root;
        for(int i = 20; i >= 0; --i) {
            int w = ((x >> i) & 1);
            if(tree[pos][w] == 0) {
                tree[pos][w] = create();
            }
            pos = tree[pos][w];
        }
    }

    // 从字典树中找到与 x 异或的最大值的结果
    int queryMax(int x) {
        int pos = root;
        int ret = 0;
        for(int i = 20; i >= 0; --i) {
            int w = ((x >> i) & 1);
            if(tree[pos][w ^ 1] == 0) {
                pos = tree[pos][w];
            } else {
                pos = tree[pos][w ^ 1];
                ret ^= (1 << i);
            }
        }
        return ret;
    }

    // 从字典树中找到与 x 异或的最小值的结果
    int queryMin(int x) {
        int pos = root;
        int ret = 0;
        for(int i = 20; i >= 0; --i) {
            int w = ((x >> i) & 1);
            if(tree[pos][w] == 0) {
                pos = tree[pos][w ^ 1];
                ret ^= (1 << i);
            } else {
                pos = tree[pos][w];
            }
        }
        return ret;
    }
} trie;

int n, ans, sum;
int num[maxn];
int lmax[maxn], rmax[maxn], lmin[maxn], rmin[maxn];

int main() {
#ifdef ExRoc
    freopen("test.txt", "r", stdin);
#endif // ExRoc
    ios::sync_with_stdio(false);

    cin >> n;

    trie.Init();
    // 前缀和必须将第 0 位放入
    sum = 0;
    trie.add(sum);
    for (int i = 1; i <= n; ++i) {
        cin >> num[i];
        sum ^= num[i];
        lmax[i] = trie.queryMax(sum);
        lmin[i] = trie.queryMin(sum);
        trie.add(sum);
    }

    trie.Init();
    sum = 0;
    trie.add(sum);
    for (int i = n; i >= 1; --i) {
        sum ^= num[i];
        rmax[i] = trie.queryMax(sum);
        rmin[i] = trie.queryMin(sum);
        trie.add(sum);
    }

    // 最后做一遍前缀 / 后缀 max / min
    for (int i = 2; i <= n; ++i) {
        lmax[i] = max(lmax[i], lmax[i - 1]);
        lmin[i] = min(lmin[i], lmin[i - 1]);
        rmax[n - i + 1] = max(rmax[n - i + 1], rmax[n - i + 2]);
        rmin[n - i + 1] = min(rmin[n - i + 1], rmin[n - i + 2]);
    }

    for (int i = 1; i < n; ++i) {
        ans = max({ans, lmax[i] - rmin[i + 1], rmax[i + 1] - lmin[i]});
    }
    cout << ans << endl;

    return 0;
}
相关推荐
_wyt0011 小时前
洛谷 B3930 [GESP202312 五级] 烹饪问题 题解
c++·gesp
通信小呆呆4 小时前
当算法有了“五感”:多模态数据融合如何向人体感官协同学习?
人工智能·学习·算法·机器学习·机器人
benben0444 小时前
强化学习之DQN算法族(基于gymnasium开发)
算法
玖玥拾5 小时前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
何以解忧,唯有..5 小时前
Go语言循环语句详解:for、range与循环控制
开发语言·算法·golang
想吃火锅10056 小时前
【leetcode】88.合并两个有序数组js
算法
один but you6 小时前
constexpr函数
c++
生成论实验室7 小时前
机器人:一个自主运动的系统
人工智能·算法·语言模型·机器人·自动驾驶·agi·安全架构
Qres8217 小时前
算法复键——树状数组
数据结构·算法