【GDKOI2016Day1T1-魔卡少女】【拆位】线段树维护区间内所有连续子区间的异或和
题意:给出N个数,M个操作。操作有修改和询问两种,每次修改将一个数改成另一个数,每次询问一个区间的所有连续子区间的异或和。n,m<=100000,ai<=1000

题解:
当年(其实也就是今年)做不出来的题。。D1T1啊。。。
因为ai<=1000,我们可以拆位处理。拆成10个二进制位,每位开1棵线段树。
对于每个节点,维护:
d:这段区间的异或和
L[0],L[1]:子区间一定从左端点开始,异或和为0,1的子区间分别有多少个
R[0],R[1]:子区间一定从右端点开始,异或和为0,1的子区间分别有多少个
s[0],s[1]:异或和为0,1的子区间分别有多少个
然后重点就是合并啦。
node upd(int ind,int tmp,node lc,node rc)
{
int dl=lc.d,dr=rc.d;
node x;
if(tmp!=) x=t[ind][tmp];
x.d=lc.d^rc.d;
x.L[]=(lc.L[]+rc.L[(dl==) ? :])%mod;
x.L[]=(lc.L[]+rc.L[(dl==) ? :])%mod;
x.R[]=(rc.R[]+lc.R[(dr==) ? :])%mod;
x.R[]=(rc.R[]+lc.R[(dr==) ? :])%mod;
x.s[]=(lc.s[]+rc.s[]+(lc.R[]*rc.L[])%mod+(lc.R[]*rc.L[])%mod)%mod;
x.s[]=(lc.s[]+rc.s[]+(lc.R[]*rc.L[])%mod+(lc.R[]*rc.L[])%mod)%mod;
return x;
}
我打成node形式。。因为最后查询的时候有多个区间也要合并。。
代码:
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<iostream>
#include<algorithm>
using namespace std; typedef long long LL;
const int N=;
const LL mod=;
struct node{
int l,r,lc,rc,d;
LL L[],R[],s[];
//L:从左开始
//R:从右开始
//s:总答案
}t[][*N];
char c[];
int n,m,tl,a[N][];
LL bit[]; node upd(int ind,int tmp,node lc,node rc)
{
int dl=lc.d,dr=rc.d;
node x;
if(tmp!=) x=t[ind][tmp];
x.d=lc.d^rc.d;
x.L[]=(lc.L[]+rc.L[(dl==) ? :])%mod;
x.L[]=(lc.L[]+rc.L[(dl==) ? :])%mod;
x.R[]=(rc.R[]+lc.R[(dr==) ? :])%mod;
x.R[]=(rc.R[]+lc.R[(dr==) ? :])%mod;
x.s[]=(lc.s[]+rc.s[]+(lc.R[]*rc.L[])%mod+(lc.R[]*rc.L[])%mod)%mod;
x.s[]=(lc.s[]+rc.s[]+(lc.R[]*rc.L[])%mod+(lc.R[]*rc.L[])%mod)%mod;
return x;
} int bt(int ind,int l,int r)
{
int x=++tl;
t[ind][x].l=l;t[ind][x].r=r;
t[ind][x].lc=t[ind][x].rc=;
t[ind][x].d=;
memset(t[ind][x].L,,sizeof(t[ind][x].L));
memset(t[ind][x].R,,sizeof(t[ind][x].R));
memset(t[ind][x].s,,sizeof(t[ind][x].s));
if(l<r)
{
int mid=(l+r)/;
t[ind][x].lc=bt(ind,l,mid);
t[ind][x].rc=bt(ind,mid+,r);
int lc=t[ind][x].lc,rc=t[ind][x].rc;
t[ind][x]=upd(ind,x,t[ind][lc],t[ind][rc]);
}
else
{
int d=a[l][ind];
t[ind][x].d=d;
t[ind][x].L[d]=t[ind][x].R[d]=t[ind][x].s[d]=;
}
return x;
} void change(int ind,int x,int p,int d)
{
if(t[ind][x].l==t[ind][x].r)
{
t[ind][x].d=d;
t[ind][x].L[d]=t[ind][x].R[d]=t[ind][x].s[d]=;
t[ind][x].L[d^]=t[ind][x].R[d^]=t[ind][x].s[d^]=;
return ;
}
int lc=t[ind][x].lc,rc=t[ind][x].rc,mid=(t[ind][x].l+t[ind][x].r)/;
if(p<=mid) change(ind,lc,p,d);
else change(ind,rc,p,d);
t[ind][x]=upd(ind,x,t[ind][lc],t[ind][rc]);
} node query(int ind,int x,int l,int r)
{
if(t[ind][x].l==l && t[ind][x].r==r) return t[ind][x];
int lc=t[ind][x].lc,rc=t[ind][x].rc,mid=(t[ind][x].l+t[ind][x].r)/;
if(r<=mid) return query(ind,lc,l,r);
else if(l>mid) return query(ind,rc,l,r);
else
{
node a0=query(ind,lc,l,mid);
node a1=query(ind,rc,mid+,r);
return upd(,,a0,a1);
}
} void output(int ind,int x)
{
int lc=t[ind][x].lc,rc=t[ind][x].rc;
printf("l=%d r=%d d=%d l0=%lld l1=%lld r0=%lld r1=%lld s0=%lld s1=%lld\n",t[ind][x].l,t[ind][x].r,t[ind][x].d,t[ind][x].L[],t[ind][x].L[],t[ind][x].R[],t[ind][x].R[],t[ind][x].s[],t[ind][x].s[]);
if(lc) output(ind,lc);
if(rc) output(ind,rc);
} int main()
{
freopen("a.in","r",stdin);
freopen("me.out","w",stdout);
// freopen("cardcaptor.in","r",stdin);
// freopen("cardcaptor.out","w",stdout);
scanf("%d",&n);
int x,ind;node now;
bit[]=;
for(int i=;i<=;i++) bit[i]=bit[i-]*;
memset(a,,sizeof(a));
for(int i=;i<=n;i++)
{
scanf("%d",&x);
ind=;
while(x)
{
a[i][ind]=x%;
x/=;
ind++;
}
}
scanf("%d",&m);
for(int i=;i<;i++) {tl=;bt(i,,n);}
for(int i=;i<=m;i++)
{
scanf("%s",c);
if(c[]=='Q')
{
int l,r;LL ans=;
scanf("%d%d",&l,&r);
for(int j=;j<;j++)
{
now=query(j,,l,r);
ans=(ans+(bit[j]*now.s[])%mod)%mod;
}
printf("%lld\n",ans);
}
else
{
int ind=,p,d;
scanf("%d%d",&p,&d);
while(d)
{
change(ind,,p,d%);
d/=;
ind++;
}
for(int j=ind;j<;j++) change(j,,p,);
}
}
return ;
}
【GDKOI2016Day1T1-魔卡少女】【拆位】线段树维护区间内所有连续子区间的异或和的更多相关文章
- POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 )
POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 ) 题意分析 给出n个点,m个询问,和当前位置pos. 先给出n-1条边,u->v以及边权w. 然后有m个询问 ...
- Can you answer these queries V SPOJ - GSS5 (分类讨论+线段树维护区间最大子段和)
recursion有一个整数序列a[n].现在recursion有m次询问,每次她想知道Max { A[i]+A[i+1]+...+A[j] ; x1 <= i <= y1 , x2 &l ...
- 线段树维护区间前k小
线段树维护区间前k小 $ solution: $ 觉得超级钢琴太麻烦?在这里线段树提供一条龙服务 . 咳咳,开始讲正题!这道题我们有一个和超级钢琴复杂度一样 $ ~O(~\sum x\times lo ...
- CodeForces - 587E[线段树+线性基+差分] ->(线段树维护区间合并线性基)
题意:给你一个数组,有两种操作,一种区间xor一个值,一个是查询区间xor的结果的种类数 做法一:对于一个给定的区间,我们可以通过求解线性基的方式求出结果的种类数,而现在只不过将其放在线树上维护区间线 ...
- hdu_5726_GCD(线段树维护区间+预处理)
题目链接:hdu_5726_GCD 题意: 给你n个数(n<=1e5)然后m个询问(m<=1e5),每个询问一个区间,问你这个区间的GCD是多少,并且输出从1到n有多少个区间的GCD和这个 ...
- FJUT3568 中二病也要敲代码(线段树维护区间连续最值)题解
题意:有一个环,有1~N编号,m次操作,将a位置的值改为b,问你这个环当前最小连续和多少(不能全取也不能不取) 思路:用线段树维护一个区间最值连续和.我们设出两个变量Lmin,Rmin,Mmin表示区 ...
- 滑动窗口(poj,线段树维护区间最值)
题目描述 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值. 例如: The array i ...
- 51nod 1376【线段树维护区间最大值】
引自:wonter巨巨的博客 定义 dp[i] := 以数字 i(不是下标 i)为结尾的最长上升长度 然后用线段树维护 dp[i]: 每个节点维护 2 个信息,一个是当前区间的最大上升长度,一个是最大 ...
- [CSP-S模拟测试]:椎(线段树维护区间最值和单调栈)
题目描述 虽不能至,心向往之. $Treap=Tree+Heap$ 椎$=$树$+$堆 小$\pi$学习了计算机科学中的数据结构$Treap$. 小$\pi$知道$Treap$指的是一种树. 小$\p ...
随机推荐
- 使用cout进行格式化
以下内容摘自木缥缈的博客 使用cout进行格式化 ostream插入运算符将值转换为文本格式.在默认情况下,格式化值的方式如下. * 对于char值,如果它代表的是可打印字符,则将被作为一个字符显示在 ...
- LintCode-70.二叉树的层次遍历 II
二叉树的层次遍历 II 给出一棵二叉树,返回其节点值从底向上的层次序遍历(按从叶节点所在层到根节点所在的层遍历,然后逐层从左往右遍历) 样例 给出一棵二叉树 {3,9,20,#,#,15,7}, 按照 ...
- Node js MySQL简单操作
//win7环境下node要先安装MySQL的相关组件(非安装MySQL数据库),在cmd命令行进入node项目目录后执行以下语句 //npm install mysql var mysql = re ...
- DEDE去掉会员登录及注册验证码的方法
1.登录打开member/index_do.php 删除245-250行,即: if(strtolower($vdcode)!=$svali || $svali=='') { ResetVdValue ...
- Directory类的使用、Alt+Shift+F10可以查看其命名空间
对于一个对象,按下Alt+Shift+F10可以查看其命名空间. Directory类的使用 using System; using System.Collections.Generic; using ...
- 钉钉 E应用 打开分享外链
钉钉 E应用 打开分享外链 外部链接 https://open-doc.dingtalk.com/microapp/dev https://open-doc.dingtalk.com/microapp ...
- Delphi Dataset CurValue
TField.CurValue Property Represents the current value of the field component including changes made ...
- 【EF Core】Entity Framework Core 批处理语句
在Entity Framework Core (EF Core)有许多新的功能,最令人期待的功能之一就是批处理语句.那么批处理语句是什么呢?批处理语句意味着它不会为每个插入/更新/删除语句发送单独的请 ...
- C++面向对象编程,继承,数据抽象,动态绑定
派生类(derived class)能够继承基类(base class )定义的成员: 1).派生类可以无需改变而使用那些与派生类具体特性不相关的操作 2).可以重新定义那些与派生类相关的成员函数,将 ...
- BZOJ 1076 奖励关(状压期望DP)
当前得分期望=(上一轮得分期望+这一轮得分)/m dp[i,j]:第i轮拿的物品方案为j的最优得分期望 如果我们正着去做,会出现从不合法状态(比如前i个根本无法达到j这种方案),所以从后向前推 如果当 ...