首页 > 其他分享 >P4839 P 哥的桶 题解

P4839 P 哥的桶 题解

时间:2023-10-02 22:12:32浏览次数:45  
标签:pr node int 题解 tree P4839 maxn ans

题目大意


\(n\)
个桶,
\(m\)
次操作。


\(pos\)
桶中加入一个
\(val\)
值,


\([l,r]\)
中选任意个桶使得异或和最大,求最大的异或和,

注意每个节点是一个桶可以放多个值 \(n,m≤5×104\) 。

题目思路

单点修改,区间查询,异或最大值

很显然是线段树维护线性基

然后这样的复杂度是
\(O(n×log3)\)

然而会
TLE

看了一下大佬的题解,更新的时候没有必要维护两个子节点的并集。

只需要从上往下更新,每到一个节点就更新其线性基维护的值就好了。

查询的时候每次都将不同区间所维护的线性基合并到答案即可。

插入复杂度
\(O(n×log2)\),

查询复杂度
\(O(n×log3)\),

但是这个查询的常数还是大大优化了。

TLE代码:

#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
typedef long long ll;
typedef pair<double,int> pdi;
const ll INF=0x3f3f3f3f3f3f3f3f;
const int maxn=5e4+5,mod=1e7;
int n,m;
int a[maxn];
int p[40];
vector<int> tree[maxn<<2];
void ins(int x) {
    for(int i = 30;i>=0; i--) {
        if (!(x &(1<<i)))  continue;
        // x的第i位是0
        if(!p[i]){
            p[i] = x;
            break;
        }
        x^= p[i];
    }
}
vector<int> mer(vector<int> a,vector<int> b){
    vector<int> ans;
    memset(p,0,sizeof(p));
    for(int i=0;i<a.size();i++) ins(a[i]);
    for(int i=0;i<b.size();i++) ins(b[i]);
    for(int i=0;i<=30;i++){
        if(p[i]!=0){
            ans.push_back(p[i]);
        }
    }
    return ans;
}
void update(int node,int pos,int l,int r,int val){
    if(l==r){
        vector<int> temp;
        temp.push_back(val);
        tree[node]=mer(tree[node],temp);
        return ;
    }
    int mid=(l+r)/2;
    if(mid>=pos) update(node<<1,pos,l,mid,val);
    else  update(node<<1|1,pos,mid+1,r,val);
    tree[node]=mer(tree[node<<1],tree[node<<1|1]);
}
vector<int> query(int node,int L,int R,int l,int r){
    if(L<=l&&r<=R){
        return tree[node];
    }
    vector<int> ans;
    int mid=(l+r)/2;
    if(mid>=L) ans=mer(ans,query(node<<1,L,R,l,mid));
    if(mid<R)  ans=mer(ans,query(node<<1|1,L,R,mid+1,r));
    return ans;
}
signed main(){
    scanf("%d%d",&m,&n);
    for(int i=1,opt,pos,val,l,r;i<=m;i++){
        scanf("%d",&opt);
        if(opt==1){
            scanf("%d%d",&pos,&val);
            update(1,pos,1,n,val);
        }else{
            scanf("%d%d",&l,&r);
            vector<int> ans=query(1,l,r,1,n);
            int pr=0;
            for(int j=(int)ans.size()-1;j>=0;j--){
                pr=max(pr,pr^ans[j]);
            }
            printf("%d\n",pr);
        }
    }
    return 0;
}

AC代码:

#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
typedef long long ll;
typedef pair<double,int> pdi;
const ll INF=0x3f3f3f3f3f3f3f3f;
const int maxn=5e4+5,mod=1e7;
int n,m;
int a[maxn];
int tree[maxn<<2][40];
void ins(int pos,int x) {
    for(int i = 30;i>=0; i--) {
        if (!(x &(1<<i)))  continue;
        // x的第i位是0
        if(!tree[pos][i]){
            tree[pos][i] = x;
            break;
        }
        x^= tree[pos][i];
    }
}
void update(int node,int pos,int l,int r,int val){
    ins(node,val);
    if(l==r){
        return ;
    }
    int mid=(l+r)/2;
    if(mid>=pos) update(node<<1,pos,l,mid,val);
    else  update(node<<1|1,pos,mid+1,r,val);
}
void query(int node,int L,int R,int l,int r){
    if(L<=l&&r<=R){
        for(int i=0;i<=30;i++){
            if(!tree[node][i]) continue;
            ins(0,tree[node][i]);
        }
        return ;
    }
    int mid=(l+r)/2;
    if(mid>=L) query(node<<1,L,R,l,mid);
    if(mid<R)  query(node<<1|1,L,R,mid+1,r);
}

signed main(){
    scanf("%d%d",&m,&n);
    for(int i=1,opt,pos,val,l,r;i<=m;i++){
        scanf("%d",&opt);
        if(opt==1){
            scanf("%d%d",&pos,&val);
            update(1,pos,1,n,val);
        }else{
            scanf("%d%d",&l,&r);
            memset(tree[0],0,sizeof(tree[0]));
            query(1,l,r,1,n);
            int pr=0;
            for(int i=30;i>=0;i--){ // tree[0]为答案
                pr=max(pr,pr^tree[0][i]);
            }
            printf("%d\n",pr);
        }
    }
    return 0;
}


标签:pr,node,int,题解,tree,P4839,maxn,ans
From: https://www.cnblogs.com/BLDramer/p/17740502.html

相关文章

  • CF906C题解
    可能更好的阅读体验大家好,我和DP有仇,所以我用猜结论的方法过了这道题。可能是这道题的一个全新思路,可能人自闭久了什么都能想出来(((upd:好像这也是官方题解思路,看来大家做题不太喜欢看CF官方题解(((首先考虑一个问题:如果这是一道构造题,怎么构造一组合法的解?在草稿纸上画了很久......
  • P2230 Tinux系统 题解
    传送门题目大意:一个\(n\)个叶子节点,一个节点最多可以有\(k\)条边连向子节点,每个节点\(i\)有一个权值\(P_{i}\)。记每个节点子树内点的个数(不包括它自己)为\(son_{i}\),那么每个节点对答案的贡献就是\(son_{i}^2\timesP_{i}\)。特别的,根节点贡献为\(0\),求总贡献。两......
  • P1045 麦森数 题解
    传送门前排提醒:本篇题解没有使用压位和快速幂,运用了一种预处理的思想,希望能提供一种新的思路。首先将\(2^{p}-1(d)\)转换为\(1111…111(b)\)。关于第一问:我们先考虑\(2\)进制转\(8\)进制,将每\(3\)位转为\(1\)位,即每\(\log{8}\)位转为\(1\)位。\(2\)进制转......
  • UVA1471 防线 Defense Lines 题解
    传送门首先可以将题意大概可以简化为:取两端不重复的连续子序列,组成一个最长的连续递增子序列。我们先dp预处理出以\(i\)为结尾的连续递增子序列长度\(dpr_{i}\)。同样预处理出以\(i\)为开头的连续递增子序列长度\(dpl_{i}\)。考虑对于每个\(dpr_{i}\),找到满足\(a_{......
  • P5503 灯塔 题解
    决策单调性二分传送门数据加强版:P3515前置知识:二分,决策单调性首先很容易写出答案式子:\[ans_{i}=\max_{j=i}^{n}{(a_{j}-a_{i}+\lceil\sqrt{\left|i-j\right|}\rceil)}\]先将向上取整符号拆掉,只要在输出时处理就行。再将绝对值符号拆掉,分\(j<i\)和\(j>i\)的情况......
  • 题解 Codeforces Round 901 (Div. 1) / CF1874A~E
    题解CodeforcesRound901(Div.1)/CF1874A~E比赛情况:过了AB。赛后发现B是假复杂度。https://codeforc.es/contest/1874A.JellyfishandGameProblemAlice&Bob又在博弈,Alice手上有\(n\)个苹果,第\(i\)个苹果的价值是\(a_i\);Bob手上有\(m\)个苹果,第\(i\)......
  • [题解]AT_abc240_f [ABC240F] Sum Sum Max
    思路题目要求的是\(\max_{a=1}^{n}\{\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\}\),所以我们将\(\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\)化简一下,得:\[i\timesA_1+(i-1)\timesA_2+\dots+1\timesA_x\]在\(a\)每增加\(1\)时,这个和\(s\)将会变为\(s+......
  • [题解]AT_abc245_f [ABC245F] Endless Walk
    思路首先我们可以发现,在任意一个节点数量大于\(1\)的强连通分量中的点都满足条件。所以,我们可以对这张图跑一边TarJan。但是这样是错的,因为我们还需要考虑节点数量为\(1\)的强连通分量。如果这种连通分量能够到达任意一个节点数量大于\(1\)的强连通分量,那么,这个连通分......
  • CSES.1141 C++题解
    题意传送门有一个长度为\(n\)的歌单,问最长多少首歌互不相同?每首歌用一个\(1-10^9\)的整数表示。样例输入812132742样例输出5算法双指针算法。桶思想。对于歌单中重复出现的数,可以用桶来存储。定义两个指针i,j,i指向大数,j指向小数。当出现某个桶的数大于1时,则......
  • CF1878C Vasilije in Cacak 题解
    题目传送门简化题意有\(t\)组询问,每次询问是否能从\(1\simn\)中选择\(k\)个数使得它们的和为\(x\)。解法考虑临界情况,从\(1\simn\)中选择最小的\(k\)个数时和为\(\sum\limits_{i=1}^ki=\dfrac{(k+1)k}{2}\),从\(1\simn\)中选择最大的\(k\)个数时和为\(......