首先我们设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. 判断一个变量是不是json,以及如何将变量转换成json

    https://blog.csdn.net/A123638/article/details/52486975这里看到一个很好的方法 // 判断变量是不是jsonisJson(variable: any ...

  2. Bare metal APIs with ASP.NET Core MVC(转)

    ASP.NET Core MVC now provides a true "one asp.net" framework that can be used for building ...

  3. 【移动端debug-2】Flexbox在移动端的兼容实践

    最近在项目中用到了flexbox,总结一下使用心得. 一.什么是flexbox,干嘛使的? 曾几何时,我们特别希望能像word一样,在排版时有个分散对齐选项(平均分配子元素宽度)这样我就可以任意在父元 ...

  4. BZOJ 1444 有趣的游戏(AC自动机+矩阵快速幂)

    真的是很有趣的游戏... 对每个单词构建好AC自动机后,由于单词都是相同长度的且不同,所以不会出现互相为子串的形式. 那么我们对AC自动机上的节点构建转移矩阵.对于每个单词末尾的节点.该节点的出边仅仅 ...

  5. 洛谷 P4116 Qtree3

    Qtree系列第三题 我是题面 读完题大概不难判断是一道树剖的题 这道题的关键是记录两种状态,以及黑点的序号(不是编号) 线段树啊当然 定义两个变量v,f,v表示距离根节点最近的黑点,默认-1,f则表 ...

  6. QoS专题-第1期-QoS理论篇

    QoS理论篇 1      QoS的产生 随着网络技术的飞速发展,IP网络已经从当初的单一数据网络向集成数据.语音.视频.游戏的多业务网络转变.网络中所承载的数据呈几何级倍数增长,而且这些业务对网络带 ...

  7. HTML option 标签的 selected 属性

    HTML option 标签的 selected 属性 http://www.w3school.com.cn/tags/att_option_selected.asp 1.一般做WEB的时候,Chro ...

  8. CF 566A Matching Names

    CF 566A Matching Names 题目描述 给出n个名字和n个昵称,求一个名字和昵称的劈配方案,使得被劈配的名字和昵称的最长公共前缀长度的和最大. 1<=n<=100000 字 ...

  9. 洛谷 P4754 True Vegetable 解题报告

    P4754 True Vegetable 题目描述 小A现在有N道题,编号为1,2,⋯,N.每道题的起始毒瘤程度为0或1.在每天,小A可以将编号连续的K道题的毒瘤程度+1.但小B因为本身比较菜,不是很 ...

  10. C++中Vector求最大值最小值

    vector<int> v: 最大值: int max = *max_element(v.begin(),v.end()); 最小值: int min = *min_element(v.b ...