首先我们设si为前i个数的xor和,那么对于询问区间[i,j]的xor和,就相当于si-1^sj,那么对于这道题的询问我们可以处理处si,然后对于询问[l,r],可以表示为在区间[l-1,r]里找两个数使得这两个数的xor值最大,对于区间中找一个数使得xor一个已知数的值最大我们可以用可持久化trie来完成(有疑问请移步http://www.cnblogs.com/BLADEVIL/p/3669219.html),设询问query(l,r,x)表示x与区间[l,r]的数的xor值最大值。那么我们可以将这n个数分为sqrt(n)个块,我们设head[i]为第i个块的最开始元素,w[i][j]为从第i个块的第一个元素开始,到第j个数的区间中,任意找两个数的xor值最大,那么易得到转移方程w[i][j]=max(w[i][j],w[i][j-1]+query(head[i],j-1,a[j]))。这个的时间复杂度是o(sqrt(n)*n*logn)的,然后对于询问[l,r],我们设l后面第一个块为x块,那么ans=max(w[x][r],max(l,r,a[j])),j∈[l,head[x]-1]。

  备注:开始的时候以为题目说a[i]不会超过32767,后来发现好像是2147483647,然后改了trie的深度之后发现总是RE,因为我写的trie是先建一个空树,然后再插入节点的,31层的满二叉树当然没有办法建出来,那么改成了直接插入,发现还是WA,因为上一次的答案可能只比2147483647小一点,那么加上这次询问的值就会超过int,所以先强制转成long long然后%n就好了。

    开始以为RE是内存的问题,所以开大了好多,懒得改了就这样吧。

/**************************************************************
    Problem: 2741
    User: BLADEVIL
    Language: C++
    Result: Accepted
    Time:8148 ms
    Memory:41680 kb
****************************************************************/
 
//By BLADEVIL
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define maxn 20000
#define maxm 400
 
using namespace std;
 
struct tree {
    int son[];
    int cnt;
    tree() {
        memset(son,,sizeof son);
        cnt=;
    }
}t[*maxn];
 
struct rec {
    int key,num,rot;
    rec() {
        key=num=rot=;
    }
}a[maxn];
 
int n,m,len,sum,tot;
int w[maxm][maxn],head[maxm];
 
void insert(int &x,int rot,int y,int dep) {
    if (!x) x=++tot;
    if (dep==-) {
        t[x].cnt=t[rot].cnt+;
        return ;
    }
    if (y&(<<dep)) {
        insert(t[x].son[],t[rot].son[],y,dep-);
        t[x].son[]=t[rot].son[];
    } else {
        insert(t[x].son[],t[rot].son[],y,dep-);
        t[x].son[]=t[rot].son[];
    }
    t[x].cnt=t[rot].cnt+;
}
 
int query(int lx,int rx,int y,int dep) {
    if (dep==-) return ;
    if (y&(<<dep)) {
        if (t[t[rx].son[]].cnt-t[t[lx].son[]].cnt)
            return (<<dep)+query(t[lx].son[],t[rx].son[],y,dep-); else
            return query(t[lx].son[],t[rx].son[],y,dep-);
    } else {
        if (t[t[rx].son[]].cnt-t[t[lx].son[]].cnt)
            return (<<dep)+query(t[lx].son[],t[rx].son[],y,dep-); else
            return query(t[lx].son[],t[rx].son[],y,dep-);
    }
}
 
int main() {
    scanf("%d%d",&n,&m);
    for (int i=;i<=n;i++) scanf("%d",&a[i].key);
    for (int i=;i<=n;i++) a[i].key^=a[i-].key;
    sum=len=sqrt(n); if (len*len<n) len++,sum=(n+len-)/len;
    //printf("%d %d\n",len,sum);
    for (int i=;i<=n;i++) if (!head[a[i].num=(i+len-)/len]) head[a[i].num]=i; head[sum+]=n+;
    //for (int i=1;i<=n;i++) printf("%d ",a[i].key); printf("\n");
    //for (int i=1;i<=sum;i++) printf("%d ",head[i]);
    for (int i=;i<=n;i++) insert(a[i].rot,a[i-].rot,a[i].key,);
    for (int i=;i<=sum;i++)
        for (int j=head[i];j<=n;j++)
            w[i][j]=max(w[i][j-],query(a[head[i]-].rot,a[j].rot,a[j].key,));
    /*
    for (int i=1;i<=sum;i++) {
        for (int j=1;j<=n;j++) printf("%d ",w[i][j]);
        printf("\n");
    }
    */
    int ans=;
    while (m--) {
        int l,r; scanf("%d%d",&l,&r); int x,y;
        x=((long long)l+ans)%n+; y=((long long)r+ans)%n+;
        l=min(x,y); r=max(x,y); ans=; l--;
        if (a[l].num==a[r].num) {
            for (int i=l;i<=r;i++) ans=max(ans,query(a[l-].rot,a[r].rot,a[i].key,));
        } else {
            ans=w[a[l].num+][r]; //printf("%d %d %d\n",ans,a[l].num+1,r);
            for (int i=l;i<=head[a[l].num+]-;i++) ans=max(ans,query(a[l-].rot,a[r].rot,a[i].key,));
        }
        printf("%d\n",ans);
    }
    return ;
}

bzoj 2741 可持久化trie的更多相关文章

  1. BZOJ - 3166 可持久化Trie 维护次大区间

    题意:给出\(a[1...n]\),找出一个连续区间\(a[l...r],r>l\),令该区间的次大值为\(a_k\),使得\(a_k⊕a_i,l≤i≤r\)最大,输出全局最优解 (这题意有点别 ...

  2. [BZOJ3261&BZOJ3166]可持久化trie树及其应用

    可持久化trie树 可持久化trie树现在想来是比较好理解的了,但却看了一个下午... 相当于对于每个状态建立一条链(或者说一棵trie),求解的时候只要让两个点按照相同的步子走然后看sum的大小关系 ...

  3. bzoj 2741: 【FOTILE模拟赛】L 分塊+可持久化trie

    2741: [FOTILE模拟赛]L Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 1116  Solved: 292[Submit][Status] ...

  4. BZOJ 2741 【FOTILE模拟赛】L(可持久化trie)

    http://www.lydsy.com/JudgeOnline/problem.php?id=2741 思路:我们先将a变成a的异或前缀,这样问题就变成了,在l-1到r区间内,找出i,j令a[i]^ ...

  5. bzoj 2741 分块+可持久化trie

    多个询问l,r,求所有子区间异或和中最大是多少 强制在线 做法: 分块+可持久化trie 1.对于每块的左端点i,预处理出i到任意一个j,()i,j)间所有子区间异或和中最大为多少,复杂度O(\(n\ ...

  6. BZOJ 2741: 【FOTILE模拟赛】L [分块 可持久化Trie]

    题意: 区间内最大连续异或和 5点调试到现在....人生无望 但总算A掉了 一开始想错可持久化trie的作用了...可持久化trie可以求一个数与一个数集(区间中的一个数)的最大异或和 做法比较明显, ...

  7. BZOJ.2741.[FOTILE模拟赛]L(分块 可持久化Trie)

    题目链接 首先记\(sum\)为前缀异或和,那么区间\(s[l,r]=sum[l-1]^{\wedge}sum[r]\).即一个区间异或和可以转为求两个数的异或和. 那么对\([l,r]\)的询问即求 ...

  8. BZOJ 2741: 【FOTILE模拟赛】L(可持久化Trie+分块)

    传送门 解题思路 首先求出前缀异或和,那么问题就转化成了区间内选两个数使得其异或和最大.数据范围不是很大考虑分块,设\(f[x][i]\)表示第\(x\)块开头到\(i\)这个位置与\(a[i]\)异 ...

  9. [BZOJ 4103] [Thu Summer Camp 2015] 异或运算 【可持久化Trie】

    题目链接:BZOJ - 4103 题目分析 THUSC滚粗之后一直没有写这道题,从来没写过可持久化Trie,发现其实和可持久化线段树都是一样的.嗯,有些东西就是明白得太晚. 首先Orz ZYF-ZYF ...

随机推荐

  1. hadoop对于压缩文件的支持

    转载:https://www.cnblogs.com/ggjucheng/archive/2012/04/22/2465580.html hadoop对于压缩格式的是透明识别,我们的MapReduce ...

  2. 【C++】指针和引用

    引用: 引用(reference)是为对象起了另外一个名字,引用类型应用(refers to)另外一种类型.通过将声明符写成&d的形式来定义引用类型,其中d是声明的变量名. 一般初始化变量时, ...

  3. 【C++】C++函数重载的总结

    函数重载: 出现在相同作用域中的两个函数,如果具有相同的名字而形参表不同,则称为重载函数(overloaded function).一定要注意函数重载的两个关键词:形参列表和作用域. 任何程序有且仅有 ...

  4. HDU3507_Print Article

    这个题目又是一个典型的dp斜率优化的题目.题意是给你n个数,你需要做的是把这个n个数分为连续的若干段,每段的权值为这段数字的和的平方加上M.求最小的总权值. 我们可以根据题意写出朴素版的dp状态转移方 ...

  5. 集合里面的 E是泛型 暂且认为是object

    集合里面的 E是泛型 暂且认为是object

  6. 包装类 integer 当做 list的参数时候 会出现无法删除成功的现象

  7. CF373C-Counting Kangaroos is Fun

    题意 有\(n\)只袋鼠,每只袋鼠有一个体积,如果一个袋鼠的体积小于等于另一个袋鼠体积的一半,那么这个袋鼠就可以被那一个袋鼠装进袋里.一个装了袋鼠的袋鼠不能再装或被装.被装进袋子的袋鼠就看不到了. 问 ...

  8. TCP的拥塞控制 (三)

    1.   Multiple Packet Losses Fast Retransmit/Fast Recovery机制可以很好地处理单个packet丢失的问题,但当大量packet同时丢包时(一个RT ...

  9. JS中数组和字符串具有的方法,以及substring,substr和slice的用法与区别

     String 对象属性 属性 描述 constructor 对创建该对象的函数的引用 length 字符串的长度 prototype 允许您向对象添加属性和方法 String 对象方法 方法 描述 ...

  10. oracle 11g用户名的大小写问题

    oracle 11g 用户名和密码默认区分大小写,可更改alter system set sec_case_sensitive_logon=false 设置改为不区分大小写.