思路
现在提供一种新的思路。用线段树。
模型
这道题的模型实际就是这样:
1 给你一段区间
2 区间上的[x,y]减去d
3 判定区间中是否有数执行完操作二后小于0,若有
输出
-1
i 。
4 若执行完所有操作后都没有出现3 中情况,输出0;
理由
因为题目中有区间操作,而线段树是一种适用于区间操作的数据结构。
线段树的基本操作如下:
1 区间加减
2 区间查询
实践
题意表明我们的目标是判定区间中是否有数小于0。
那么我们可以维护一个区间最小值。完美解决问题。
本人代码:
`#include<bits/stdc++.h>
#define ri register int
using namespace std;
const int ll=1000010;
struct no{
long long l,r,data;
}t[ll<<2];
long long n,m,a[ll],lz[ll<<2],flag=0;
inline void build(ri p,ri l,ri r){
t[p].l=l; t[p].r=r;
if(l==r){
t[p].data=a[l];
return ;
}
ri mid=(l+r)>>1;
build(p<<1,l,mid); build(p<<1|1,mid+1,r);
t[p].data=min(t[p<<1].data,t[p<<1|1].data);
}
inline void push_down(ri node){
if(lz[node]){
int l=node<<1,r=node<<1|1;
lz[l]+=lz[node];
lz[r]+=lz[node];
t[l].data+=lz[node];
t[r].data+=lz[node];
lz[node]=0;
}
}
inline void change(ri p,ri l,ri r,ri add){
if(l<=t[p].l&&r>=t[p].r){
t[p].data+=add;
lz[p]+=add;
return ;
}
push_down(p);
ri mid=(t[p].l+t[p].r)>>1;
if(l<=mid) change(p<<1,l,r,add);
if(r>mid) change(p<<1|1,l,r,add);
t[p].data=min(t[p<<1].data,t[p<<1|1].data);
}
inline long long ask(ri p,ri l,ri r){
if(l<=t[p].l&&r>=t[p].r){
return t[p].data;
}
push_down(p);
ri mid=(t[p].l+t[p].r)>>1;
long long ans=1<<30;
if(l<=mid) ans=min(ans,ask(p<<1,l,r));
if(r>mid) ans=min(ans,ask(p<<1|1,l,r));
return ans;
}
int main(){
scanf("%d%d",&n,&m);
for(ri i=1;i<=n;i++)
scanf("%d",a+i);
build(1,1,n);
for(ri i=1;i<=m;i++){
int k,x,y;
scanf("%d%d%d",&k,&x,&y);
if(flag==0)
change(1,x,y,-k);
if(ask(1,x,y)<0){
printf("-1\n%d",i);
return 0;
}
}
cout<<0;
return 0;
}
`