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。

题解:

公差 $k$ 和区间长度都是固定的,所以等差数列也是唯一确定的.
这里分几个情况:
$l=r$,无限制条件.
$k=0$,所有数必须都相同.
其他情况,令区间最大值为 $Max$,最小值为 $Min$,那么 $Max-Min=(len-1)\times k$
除此之外,还需满足没有数字重复出现,任意两个数的差都是 $k$ 的倍数 
#include <cstdio>
#include <string>
#include <cstring>
#include <algorithm>
#include <map>
#include <set>
#define ll long long
using namespace std;
namespace IO {
#define rint register int
void setIO(string s) {
string in=s+".in",out=s+".out";
freopen(in.c_str(),"r",stdin);
freopen(out.c_str(),"w",stdout);
}
inline int read(){
rint x=0,f=1;char ch=getchar();
while (ch<'0' || ch>'9'){if (ch=='-')f=-1;ch=getchar();}
while ('0'<=ch && ch<='9')x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return x*f;
}
};
const int maxn=300006;
const int inf=1000000002;
int n,m;
int arr[maxn],brr[maxn];
int GCD(int a,int b) {
return b?GCD(b,a%b):a;
}
namespace A {
int minv[maxn<<2],maxv[maxn<<2];
void pushup(int l,int r,int now) {
int mid=(l+r)>>1;
minv[now]=minv[now<<1];
maxv[now]=maxv[now<<1];
if(r>mid) {
minv[now]=min(minv[now],minv[now<<1|1]);
maxv[now]=max(maxv[now],maxv[now<<1|1]);
}
}
void build(int l,int r,int now) {
if(l==r) {
minv[now]=maxv[now]=arr[l];
return;
}
int mid=(l+r)>>1;
build(l,mid,now<<1);
if(r>mid) build(mid+1,r,now<<1|1);
pushup(l,r,now);
}
void update(int l,int r,int now,int p,int v) {
if(l==r) {
minv[now]=maxv[now]=v;
return;
}
int mid=(l+r)>>1;
if(p<=mid) update(l,mid,now<<1,p,v);
else update(mid+1,r,now<<1|1,p,v);
pushup(l,r,now);
}
int qmax(int l,int r,int now,int L,int R) {
if(l>=L&&r<=R) return maxv[now];
int mid=(l+r)>>1,re=0;
if(mid>=L) re=max(re,qmax(l,mid,now<<1,L,R));
if(R>mid) re=max(re,qmax(mid+1,r,now<<1|1,L,R));
return re;
}
int qmin(int l,int r,int now,int L,int R) {
if(l>=L&&r<=R) return minv[now];
int mid=(l+r)>>1,re=inf;
if(mid>=L) re=min(re,qmin(l,mid,now<<1,L,R));
if(R>mid) re=min(re,qmin(mid+1,r,now<<1|1,L,R));
return re;
}
};
namespace lst {
int id;
int lst[maxn<<2];
map<int,int>idx;
set<int>S[maxn<<1];
set<int>::iterator it;
void insert(int v,int i) {
if(!idx[v]) idx[v]=++id;
int cur=idx[v];
S[cur].insert(i);
}
void pushup(int l,int r,int now) {
int mid=(l+r)>>1;
lst[now]=lst[now<<1];
if(r>mid) lst[now]=max(lst[now],lst[now<<1|1]);
}
void build(int l,int r,int now) {
if(l==r) {
int cur=idx[arr[l]];
it=S[cur].lower_bound(l);
if(it==S[cur].begin()) lst[now]=-inf;
else {
it--;
lst[now]=(*it);
}
return;
}
int mid=(l+r)>>1;
if(l<=mid) build(l,mid,now<<1);
if(r>mid) build(mid+1,r,now<<1|1);
pushup(l,r,now);
}
void update(int l,int r,int now,int p) {
if(l==r) {
int cur=idx[arr[l]];
it=S[cur].lower_bound(l);
if(it==S[cur].begin()) lst[now]=-inf;
else {
it--;
lst[now]=(*it);
}
return;
}
int mid=(l+r)>>1;
if(p<=mid) update(l,mid,now<<1,p);
else update(mid+1,r,now<<1|1,p);
pushup(l,r,now);
}
int query(int l,int r,int now,int L,int R) {
if(l>=L&&r<=R) return lst[now];
int mid=(l+r)>>1,re=0;
if(L<=mid) re=max(re, query(l,mid,now<<1,L,R));
if(R>mid) re=max(re, query(mid+1,r,now<<1|1,L,R));
return re;
}
void modify(int v,int i) {
if(!idx[v]) idx[v]=++id;
int cur=idx[v];
S[idx[arr[i]]].erase(i);
it=S[idx[arr[i]]].lower_bound(i+1);
if(it!=S[idx[arr[i]]].end()) {
update(1,n,1,(*it));
}
S[cur].insert(i), arr[i]=v;
update(1,n,1,i);
}
};
namespace delta {
int gc[maxn<<2];
void pushup(int l,int r,int now) {
int mid=(l+r)>>1;
gc[now]=gc[now<<1];
if(r>mid) gc[now]=GCD(gc[now<<1|1],gc[now]);
}
void build(int l,int r,int now) {
if(l==r) {
gc[now]=brr[l];
return;
}
int mid=(l+r)>>1;
if(l<=mid) build(l,mid,now<<1);
if(r>mid) build(mid+1,r,now<<1|1);
pushup(l,r,now);
}
void update(int l,int r,int now,int p,int v) {
if(l==r) {
gc[now]=v;
return;
}
int mid=(l+r)>>1;
if(p<=mid) update(l,mid,now<<1,p,v);
else update(mid+1,r,now<<1|1,p,v);
pushup(l,r,now);
}
int query(int l,int r,int now,int L,int R) {
if(l>=L&&r<=R) return gc[now];
int mid=(l+r)>>1,re=0;
if(L<=mid) re=query(l,mid,now<<1,L,R);
if(R>mid) re=re?GCD(re,query(mid+1,r,now<<1|1,L,R)):query(mid+1,r,now<<1|1,L,R);
return re;
}
};
int main() {
using namespace IO;
// IO::setIO("input");
int i,j;
n=read(),m=read();
for(i=1;i<=n;++i) {
arr[i]=read();
lst::insert(arr[i],i);
}
for(i=1;i<=n;++i) brr[i]=abs(arr[i+1]-arr[i]);
A::build(1,n,1);
delta::build(1,n-1,1);
lst::build(1,n,1);
int cc=0;
int h=0;
for(int cas=1;cas<=m;++cas) {
int op=read();
if(op==1) {
int x,y;
x=read(),y=read(),x^=cc,y^=cc;
if(arr[x]==y) continue;
lst::modify(y,x);
A::update(1,n,1,x,y);
if(x==1) delta::update(1,n-1,1,1,abs(arr[2]-arr[1]));
else if(x==n) delta::update(1,n-1,1,n-1,abs(arr[n]-arr[n-1]));
else {
delta::update(1,n-1,1,x-1,abs(arr[x]-arr[x-1]));
delta::update(1,n-1,1,x,abs(arr[x+1]-arr[x]));
}
}
if(op==2) {
int l,r,k;
l=read(),r=read(),k=read();
l^=cc,r^=cc,k^=cc;
if(l==r) printf("Yes\n"),++cc;
else if(k==0) {
if(A::qmin(1,n,1,l,r)==A::qmax(1,n,1,l,r)) printf("Yes\n"),++cc;
else printf("No\n");
}
else {
int len=r-l;
int minn=A::qmin(1,n,1,l,r), maxx=A::qmax(1,n,1,l,r);
int flag=0;
if(maxx-minn==1ll*k*len) {
if(lst::query(1,n,1,l,r)>=l) flag=1;
else {
if(delta::query(1,n-1,1,l,r-1)%k) flag=1;
else {
printf("Yes\n"), ++cc;
}
}
}
else flag=1;
if(flag) printf("No\n");
}
}
}
return 0;
}

  

BZOJ 4373: 算术天才⑨与等差数列 线段树的更多相关文章

  1. bzoj 4373 算术天才⑨与等差数列——线段树+set

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4373 能形成公差为k的等差数列的条件:mx-mn=k*(r-l) && 差分 ...

  2. BZOJ 4373 算术天才⑨与等差数列 线段树+set(恶心死我了)

    mdzz,这道题重构了4遍,花了一个晚上... 满足等差数列的条件: 1. 假设min是区间最小值,max是区间最大值,那么 max-min+k(r−l) 2. 区间相邻两个数之差的绝对值的gcd=k ...

  3. BZOJ 4373算术天才⑨与等差数列(线段树)

    题意:给你一个长度为n的序列,有m个操作,写一个程序支持以下两个操作: 1. 修改一个值 2. 给出三个数l,r,k, 询问:如果把区间[l,r]的数从小到大排序,能否形成公差为k的等差数列. n,m ...

  4. bzoj 4373 算术天才⑨与等差数列

    4373: 算术天才⑨与等差数列 Time Limit: 10 Sec  Memory Limit: 128 MBhttp://www.lydsy.com/JudgeOnline/problem.ph ...

  5. 【BZOJ4373】算术天才⑨与等差数列 [线段树]

    算术天才⑨与等差数列 Time Limit: 10 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description 算术天才⑨非常喜欢和等 ...

  6. 【BZOJ4373】算术天才⑨与等差数列 线段树+set

    [BZOJ4373]算术天才⑨与等差数列 Description 算术天才⑨非常喜欢和等差数列玩耍.有一天,他给了你一个长度为n的序列,其中第i个数为a[i].他想考考你,每次他会给出询问l,r,k, ...

  7. bzoj 4373: 算术天才⑨与等差数列 hash

    题目链接 题目大意:  给你n个数, 给两种操作, 一种给你l, r, k,问你[l, r]区间里的数排序后能否构成一个公差为k的等差数列. 另一种是将位置x的数变为y. 强制在线. 可以用hash来 ...

  8. [BZOJ4373]算术天才⑨与等差数列(线段树)

    [l,r]中所有数排序后能构成公差为k的等差数列,当且仅当: 1.区间中最大数-最小数=k*(r-l) 2.k能整除区间中任意两个相邻数之差,即k | gcd(a[l+1]-a[l],a[l+2]-a ...

  9. bzoj4373 算术天才⑨与等差数列——线段树+set

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4373 一个区间有以 k 为公差的数列,有3个条件: 1.区间 mx - mn = (r-l) ...

随机推荐

  1. 20191128 Spring Boot官方文档学习(10)

    10.附录 附录A:通用应用程序属性 附录B:配置元数据 附录C:自动配置类 附录D:测试的自动配置注释 附录E:可执行的Jar格式 附录F:依赖版本

  2. 数据库SQL语句查询指定时间段内的数据

    [摘要]有的时候,我们需要查询数据库某段时间之间的数据,比如2016年5月1号到到5月3号之间用户注册数量(特殊节假日期间)等.那么用SQL语句如何实现呢? 首先,数据表中的存时间的字段比如是addt ...

  3. mac搭建apace和php开发环境

    启动Apache   1 先介绍几个命令 // 启动Apache服务 sudo apachectl start // 重启Apache服务 sudo apachectl restart // 停止Ap ...

  4. PostgreSQL通过解析日志,获取数据库增量变化,pg_recvlogical

    1.首先用该工具来看我们的日志变化,需要先将test_decoding插件编译并安装(进入contrib,编译安装即可) 创建一个slot: SELECT * FROM pg_create_logic ...

  5. 小记------查看‘阿里云机器’yarn 日志

    通过ip:8088 页面 复制正在运行的application ID    在linux客户端执行 xshell yarn logs -applicationId application_155869 ...

  6. 小记-----一些linux操作小操作

    lrzsz工具 window系统与linux系统 文件互传      1.在linux系统命令行:sudo yum install lrzsz  或者  yum install lrzsz (输入一个 ...

  7. 洛谷 P1134 阶乘问题 题解

    题面 很裸的边取模边乘.注意因为进位的原因模数应该比较大: 另外,这道题是一道标准的分块打表例题(那样的话数据就可以更大了),可以用来练习分块打表: #include<bits/stdc++.h ...

  8. Luogu P4438 [HNOI/AHOI2018]道路

    题目 注意到\(n\)不大并且深度不大. 记\((u,ls_u)\)为\(L\)边,\((u,rs_u)\)为\(r\)边. 所以我们可以设\(f_{p,i,j}\)表示从根到\(p\)有\(i\)条 ...

  9. golang 反射中调用方法

    反射中调用函数 众所周知,golang中的函数是可以像普通的int.float等类型变量那样作为值的,例如: package main import "fmt" func hell ...

  10. DAX/PowerBI系列 - 累计总计(Cumulative Total, Running Total)

    DAX/PowerBI系列 - 累计总计(Cumulative Total) 2017/07/23 更新:B列公式(见最后) 2019/08/08 更新:在可视化数据的时候,一定要选择日期维度的日期列 ...