bzoj 4303 数列
bzoj 4303 数列
- 二维 \(KD-Tree\) 模板题.
- \(KD-Tree\) 虽然在更新和查询的方式上类似于线段树,但其本身定义是类似于用 \(splay/fhq\ treap\) 维护区间的二叉搜索树,没有加点删点,建树时将它建成平衡的就好了.
- 这使得一个 \(node\) 的左子树管辖 \([l,mid-1]\) ,右子树管辖 \([mid+1,r]\) , \(mid\) 处的信息存在自己处,不要写混.
- 对于 \(k\) 维的 \(KD-Tree\) ,它每次更新/查询的时间复杂度是 \(O(n^{1-\frac 1 k})\) .所以本题总时间复杂度为 \(O(m\sqrt n)\).
- 在常数优化上有一个小技巧:若答案 \(ans\) 对 \(2^k\) 取模,可以直接输出 $ans&(P-1) $.本题 \(536870912=2^{29}\).
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define mp make_pair
#define pii pair<int,int>
#define inf 0x7f7f7f7f
inline int read()
{
int x=0;
bool pos=1;
char ch=getchar();
for(;!isdigit(ch);ch=getchar())
if(ch=='-')
pos=0;
for(;isdigit(ch);ch=getchar())
x=x*10+ch-'0';
return pos?x:-x;
}
const int MAXN=5e4+10;
const int P=536870912;
inline int add(int a,int b)
{
return (a + b) ;
}
inline int mul(int a,int b)
{
return a * b ;
}
int n,m,Tp;
struct node{
int tp;//维护的维度
int mi[2],ma[2];//[x/y]的最小,最大值
int v[2];//坐标
int ls,rs;
int tagadd,tagmul;
int val,len,sum;
bool operator < (const node& rhs) const
{
return v[Tp]<rhs.v[Tp];//当前比较的维度,全局变量储存.
}
node()
{
ls=rs=0;
tagadd=0;
tagmul=1;
sum=0;
}
}Tree[MAXN];
#define root Tree[o]
#define lson Tree[Tree[o].ls]
#define rson Tree[Tree[o].rs]
void pushup(int o)
{
root.mi[0]=min(root.v[0],min(lson.mi[0],rson.mi[0]));
root.mi[1]=min(root.v[1],min(lson.mi[1],rson.mi[1]));
root.ma[0]=max(root.v[0],max(lson.ma[0],rson.ma[0]));
root.ma[1]=max(root.v[1],max(lson.ma[1],rson.ma[1]));
}
int BuildTree(int l,int r,int tp)
{
Tp=tp;
int mid=(l+r)>>1;
int o=mid;
nth_element(Tree+l,Tree+mid,Tree+r+1);
root.tp=tp;
root.len=r-l+1;
if(l<mid)
root.ls=BuildTree(l,mid-1,(tp+1)%2);
if(r>mid)
root.rs=BuildTree(mid+1,r,(tp+1)%2);
pushup(o);
return o;
}
void Modifiy_mul(int o,int mulv)
{
root.val=mul(root.val,mulv);
root.tagmul=mul(root.tagmul,mulv);
root.tagadd=mul(root.tagadd,mulv);
root.sum=mul(root.sum,mulv);
}
void Modifiy_add(int o,int addv)
{
root.val=add(root.val,addv);
root.tagadd=add(root.tagadd,addv);
root.sum=add(root.sum,mul(root.len,addv));
}
void pushdown(int o)
{
if(root.tagmul!=1)
{
Modifiy_mul(root.ls,root.tagmul);
Modifiy_mul(root.rs,root.tagmul);
root.tagmul=1;
}
if(root.tagadd!=0)
{
Modifiy_add(root.ls,root.tagadd);
Modifiy_add(root.rs,root.tagadd);
root.tagadd=0;
}
}
void update(int o,int L,int R,int mulv,int addv)//修改第Tp维
{
if(L>root.ma[Tp] || R<root.mi[Tp])
return;
if(L<=root.mi[Tp] && root.ma[Tp]<=R)
{
Modifiy_mul(o,mulv);
Modifiy_add(o,addv);
return;
}
pushdown(o);
if(L<=root.v[Tp] && root.v[Tp]<=R)
{
root.val=mul(root.val,mulv);
root.val=add(root.val,addv);
}
pushdown(o);
update(root.ls,L,R,mulv,addv);
update(root.rs,L,R,mulv,addv);
root.sum=add(lson.sum,rson.sum);
root.sum=add(root.sum,root.val);
}
int query(int o,int L,int R)
{
if(L>root.ma[Tp] || R<root.mi[Tp])
return 0;
if(L<=root.mi[Tp] && root.ma[Tp]<=R)
return root.sum;
pushdown(o);
int res=0;
if(root.ls)
res=add(res,query(root.ls,L,R));
if(root.rs)
res=add(res,query(root.rs,L,R));
if(L<=root.v[Tp] && root.v[Tp]<=R)
res=add(res,root.val);
return res;
}
int rt;
void init()
{
Tree[0].ma[0]=Tree[0].ma[1]=-inf;
Tree[0].mi[0]=Tree[0].mi[1]=inf;
for(int i=1;i<=n;++i)
{
Tree[i].v[0]=i;
Tree[i].v[1]=read();
}
rt=BuildTree(1,n,0);
}
int main()
{
n=read(),m=read();
init();
while(m--)
{
int op=read();
int l,r,x,y;
if(op==0)
{
l=read(),r=read(),x=read(),y=read();
Tp=0;
update(rt,l,r,x,y);
}
else if(op==1)
{
l=read(),r=read(),x=read(),y=read();
Tp=1;
update(rt,l,r,x,y);
}
else if(op==2)
{
l=read(),r=read();
Tp=0;
printf("%d\n",query(rt,l,r)&(P-1));
}
else
{
l=read(),r=read();
Tp=1;
printf("%d\n",query(rt,l,r)&(P-1));
}
}
return 0;
}
bzoj 4303 数列的更多相关文章
- [BZOJ 2989]数列(二进制分组+主席树)
[BZOJ 2989]数列(二进制分组+主席树) 题面 给定一个长度为n的正整数数列a[i]. 定义2个位置的graze值为两者位置差与数值差的和,即graze(x,y)=|x-y|+|a[x]-a[ ...
- [BZOJ 2989]数列(CDQ 分治+曼哈顿距离与切比雪夫距离的转化)
[BZOJ 2989]数列(CDQ 分治) 题面 给定一个长度为n的正整数数列a[i]. 定义2个位置的graze值为两者位置差与数值差的和,即graze(x,y)=|x-y|+|a[x]-a[y]| ...
- BZOJ 3142 数列(组合)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=3142 题意:给出n,K,m,p.求有多少长度为K的序列A,满足:(1)首项为正整数:(2 ...
- BZOJ 4305: 数列的GCD( 数论 )
对于d, 记{ai}中是d的倍数的数的个数为c, 那么有: 直接计算即可,复杂度O(NlogN+MlogM) --------------------------------------------- ...
- bzoj 4305 数列的GCD
LINK:数列的GCD 题意: 给出一个长度为N的数列{a[n]},1<=a[i]<=M(1<=i<=N). 现在问题是,对于1到M的每个整数d,有多少个不同的数列b[1], ...
- bzoj 2989: 数列
LINK:数列 需要动一点脑子 考虑查询 暴力显然不行 考虑把绝对值拆开. 当x<=y ax<=ay时 有 y-x+ay-ax<=k x+ax>=y+ay-k 可以发现在满足前 ...
- BZOJ 2989: 数列/4170: 极光
题解: n倍经验题 首先比较容易想到的是对绝对值分类讨论 然后是4维偏序 1.查询和修改顺序 2.x>y 3.a[x]>a[y] 4.(x+a[x])-(y+a[y])<=k 这样是 ...
- 解题:BZOJ 2989 数列
题面 学习二进制分组 题目本身可以看成二维平面上的问题,转成切比雪夫距离后就是矩形和了 二进制分组是将每个修改添加到末尾,然后从后往前二进制下进位合并,这样最多同时有$\log n$组,每个修改只会被 ...
- BZOJ #2989. 数列 [树套树]
考虑转化问题模型,这个没必要可持久化,直接加点就可以了,还不用删点 每次的问题是求 曼哈顿距离,变成切比雪夫距离然后求解 然后我们考虑将这玩意旋转 45度, 然后原坐标的 \((x,y)\) 会变成 ...
随机推荐
- SonarQube介绍及部署
SonarQube介绍及部署 编写人:罗旭成 编写时间:2014-5-28 SonarQube简介 SonarQube是一个用于代码质量管理的开源平台(Java开发),用于管理源代码的质量,可以从七个 ...
- GridControl 史上最全的资料(一)
GridControl详解(一)原汁原味的表格展示 Dev控件中的表格控件GridControl控件非常强大.不过,一些细枝末节的地方有时候用起来不好找挺讨厌的.使用过程中,多半借助Demo和英文帮助 ...
- RabbitMQ 的路由模式 Topic模式
模型 生产者 package cn.wh; import java.io.IOException; import java.util.concurrent.TimeoutException; impo ...
- 推荐给开发者的11个PHP框架(转)
PHP框架对于Web开发者来说是非常有用的工具,它可以帮助使用者更快.更容易的完成项目.根据调查,PHP仍是Web开发中最受欢迎和最实用的平台之一.当谈及Web开发时,很多人依然会选择使用PHP框架, ...
- 【三小时学会Kubernetes!(四) 】Deployment实践
Deployment 部署 Kubernetes 部署可以帮助每一个应用程序的生命都保持相同的一点:那就是变化.此外,只有挂掉的应用程序才会一尘不变,否则,新的需求会源源不断地涌现,更多代码会被开发出 ...
- MongoDB 查看所有用户账号信息
在 MongoDB 中创建了很多帐号,怎么查看所有帐号信息? 1. 查看全局所有账户 2. 查看当前库下的账户 查看全局所有账户 : > use admin switched to db adm ...
- flask学习(七):URL反转
1. 什么叫反转URL:从视图函数到url的转换叫做反转url 2. 反转url的用处: 1) 在页面重定向的时候,会使用url反转 2) 在模板中,也会使用url反转 3. 实例: 打印出了url
- 第十天 1-9 rhel7-文件的归档和压缩
大纲:文件的归档和压缩1.tar命令的使用及参数解析tar.gz.bz/bz2文件的创建.查看及解压zip/unzip命令的使用 一.文件的归档和压缩 在我们的计算机中,经常会遇到有好多文件名相似或作 ...
- 简明 Nginx Location Url 配置笔记
基本配置 为了探究nginx的url配置规则,当然需要安装nginx.我使用了vagrant创建了一个虚拟环境的ubuntu,通过apt-get安装nginx.这样就不会污染mac的软件环境.通过vr ...
- day36 爬虫+http请求+高性能
爬虫 参考博客:http://www.cnblogs.com/wupeiqi/articles/5354900.html http://www.cnblogs.com/wupeiqi/articles ...