码迷,mamicode.com
首页 > 其他好文 > 详细

BZOJ 3261 最大异或和

时间:2016-05-27 21:36:34      阅读:128      评论:0      收藏:0      [点我收藏+]

标签:

可持久化trie。

一个重要的思想是前缀和。

然后SB错误调了一晚上。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 600500
using namespace std;
int n,m,x,y,z,tot=0,tree[maxn*25][3],sum[maxn*25],root[maxn];
int a[maxn],b[maxn],bin[30];
char type[4];
void gettable()
{
    bin[0]=1;
    for (int i=1;i<=24;i++)
        bin[i]=bin[i-1]<<1;
}
void insert(int bitt,int last,int &now,int x)
{
    now=++tot;
    tree[now][0]=tree[last][0];tree[now][1]=tree[last][1];
    sum[now]=sum[last]+1;
    if (bitt==-1) return;
    int tmp=x&bin[bitt];tmp>>=bitt;
    insert(bitt-1,tree[last][tmp],tree[now][tmp],x);
}
int query(int bitt,int last,int now,int x)
{
    if (bitt==-1) return 0;
    int tmp=x&bin[bitt];tmp>>=bitt;
    if (sum[tree[now][tmp^1]]-sum[tree[last][tmp^1]])
        return query(bitt-1,tree[last][tmp^1],tree[now][tmp^1],x)+bin[bitt];
    else return query(bitt-1,tree[last][tmp],tree[now][tmp],x);
}
void work1()
{
    n++;
    scanf("%d",&x);
    a[n]=x;b[n]=b[n-1]^x;
    insert(23,root[n-1],root[n],b[n]);
}
void work2()
{
    scanf("%d%d%d",&x,&y,&z);
    printf("%d\n",query(23,root[x-1],root[y],b[n]^z));
}
int main()
{
    gettable();
    scanf("%d%d",&n,&m);
    n++;a[1]=0;b[1]=0;
    for (int i=2;i<=n;i++)
    {
        scanf("%d",&a[i]);
        b[i]=b[i-1]^a[i];
    }    
    for (int i=1;i<=n;i++)
        insert(23,root[i-1],root[i],b[i]);
    for (int i=1;i<=m;i++)
    {
        scanf("%s",type);
        if (type[0]==A) work1();
        else work2();
    }
    return 0;
}

 

BZOJ 3261 最大异或和

标签:

原文地址:http://www.cnblogs.com/ziliuziliu/p/5535855.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!