博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
bzoj 4373 算术天才⑨与等差数列
阅读量:6253 次
发布时间:2019-06-22

本文共 3489 字,大约阅读时间需要 11 分钟。

4373: 算术天才⑨与等差数列

Time Limit: 10 Sec  Memory Limit: 128 MB

Description

算术天才⑨非常喜欢和等差数列玩耍。

有一天,他给了你一个长度为n的序列,其中第i个数为a[i]。
他想考考你,每次他会给出询问l,r,k,问区间[l,r]内的数从小到大排序后能否形成公差为k的等差数列。
当然,他还会不断修改其中的某一项。
为了不被他鄙视,你必须要快速并正确地回答完所有问题。
注意:只有一个数的数列也是等差数列。

Input

第一行包含两个正整数n,m(1<=n,m<=300000),分别表示序列的长度和操作的次数。

第二行包含n个整数,依次表示序列中的每个数a[i](0<=a[i]<=10^9)。
接下来m行,每行一开始为一个数op,
若op=1,则接下来两个整数x,y(1<=x<=n,0<=y<=10^9),表示把a[x]修改为y。
若op=2,则接下来三个整数l,r,k(1<=l<=r<=n,0<=k<=10^9),表示一个询问。
在本题中,x,y,l,r,k都是经过加密的,都需要异或你之前输出的Yes的个数来进行解密。

Output

输出若干行,对于每个询问,如果可以形成等差数列,那么输出Yes,否则输出No。

Sample Input

5 3
1 3 2 5 6
2 1 5 1
1 5 4
2 1 5 1

Sample Output

No
Yes
 
区间是等差数列的条件:
1、区间内差分的gcd=公差
2、区间最大值-最小值=(区间长度-1)*公差
3、如果公差不等于0,区间内没有重复的数
条件3好像要记录这一个数上一次出现的位置,很麻烦
没有管条件3,竟然A了
 
线段树维护区间最大值,最小值,gcd即可
#include
#include
#define N 300001using namespace std;int n,m,tmp,x,p;int opl,opr,w;int g,big,small;int a[N];struct TREE{ struct node { int l,r; int maxn,minn,gcd; }tr[N*4]; int get_gcd(int a,int b) { return !b ? a : get_gcd(b,a%b); } int read() { int x=0,f=1; char c=getchar(); while(c<'0'||c>'9') { if(c=='-') f=-1; c=getchar(); } while(c>='0'&&c<='9') { x=x*10+c-'0'; c=getchar(); } return x*f; } void up(int k) { tr[k].gcd=get_gcd(tr[k<<1].gcd,tr[k<<1|1].gcd); tr[k].maxn=max(tr[k<<1].maxn,tr[k<<1|1].maxn); tr[k].minn=min(tr[k<<1].minn,tr[k<<1|1].minn); } void build(int k,int l,int r) { tr[k].l=l; tr[k].r=r; if(l==r) { a[l]=read(); tr[k].maxn=tr[k].minn=a[l]; tr[k].gcd=a[l]-a[l-1]; return; } int mid=l+r>>1; build(k<<1,l,mid); build(k<<1|1,mid+1,r); up(k); } void solve(int k) { if(tr[k].l>=opl&&tr[k].r<=opr) { if(x==2) { if(p==1) g=get_gcd(g,tr[k].gcd); else { big=max(big,tr[k].maxn); small=min(small,tr[k].minn); } } else { if(p==1) { tr[k].minn=tr[k].maxn=w; a[tr[k].l]=w; tr[k].gcd=a[tr[k].l]-a[tr[k].l-1]; } else tr[k].gcd=a[tr[k].l]-a[tr[k].l-1]; } return; } int mid=tr[k].l+tr[k].r>>1; if(opl<=mid) solve(k<<1); if(opr>mid) solve(k<<1|1); if(x==1) up(k); }}tree;int main(){ n=tree.read(); m=tree.read(); tree.build(1,1,n); while(m--) { scanf("%d",&x); if(x==2) { big=-1; small=2e9; g=0; opl=tree.read(); opr=tree.read(); w=tree.read(); opl^=tmp; opr^=tmp; w^=tmp; if(opl==opr) { puts("Yes"); tmp++; continue;} opl++; p=1; tree.solve(1); opl--; p=2; tree.solve(1); if(g<0) g=-g; if(g==w&&(opr-opl)*w==(big-small)) { puts("Yes"); tmp++; } else puts("No"); } else { opl=tree.read(); w=tree.read(); opl^=tmp; w^=tmp; opr=opl; p=1; tree.solve(1); if(opr!=n) { opl++; opr++; p=2; tree.solve(1); } } }}

无限RE,原因:

1、如果点i记录的是i与i-1的差,查询区间[l,r]的差分的gcd应该查询区间[l+1,r],所以要特判l==r

2、修改点i,改了i点的差分,也改了点i+1的差分

 

转载于:https://www.cnblogs.com/TheRoadToTheGold/p/6817045.html

你可能感兴趣的文章
Delphi中获取Unix时间戳及注意事项
查看>>
8月5日起OCP电子证书正式推行
查看>>
【原创】DataNode源码演绎 第一回
查看>>
垃圾回收概念与算法
查看>>
JDBC读取MySQL的BLOB类型
查看>>
IconFont 图标svg
查看>>
TFS实现需求工作项自动级联保存
查看>>
crontab定时执行任务
查看>>
LDAP 设置指南(转)
查看>>
软件开发--开发中的辅助工具
查看>>
mysql查询今天、昨天、7天、近30天、本月、上一月 数据
查看>>
单臂路由
查看>>
大数据工程师微职位学习分享
查看>>
企业使用云服务器的优势
查看>>
dubbo Servlet Bridge Server时同时支持hessian和webservice
查看>>
lanmp一键安装包安装说明(包括lamp,lnmp,lnamp安装)
查看>>
Shell命令-文件及内容处理之head、tail
查看>>
Android碎碎念 -- 视频播放器
查看>>
关于51单片机“外部中断触发方式”的经验总结
查看>>
创建 floating IP - 每天5分钟玩转 OpenStack(106)
查看>>