树状数组维护区间加区间和
我不会线段树,别让我打线段树
#include<bits/stdc++.h>
#define ll long long
using namespace std;
mt19937 myrand(time(0));
inline ll read(){
ll x=0,w=1;
char ch=0;
while(ch<'0'||ch>'9'){
if(ch=='-')w=-1;
ch=getchar();
}
while(ch>='0'&&ch<='9'){
x=(x<<3)+(x<<1)+(ch^48);
ch=getchar();
}
return x*w;
}
void write(ll x){
if(x<0){
putchar('-');
x=-x;
}
static int sta[35];
int top=0;
do{
sta[top++]=x%10,x/=10;
}while(x);
while(top)putchar(sta[--top]+'0');
}
ll n,q,t1[200005],t2[200005],a[200005],sum,cnt,W;
inline int lowbit(int x){return x&(-x);}
void _add(int x,ll k){
ll v=x*k;
while(x<=n){
t1[x]+=k;
t2[x]+=v;
x+=lowbit(x);
}
}
void add(int l,int r,ll k){
_add(l,k);
_add(r+1,-k);
}
ll Qsum(ll *t,ll x){
ll ret=0;
while(x){
ret+=t[x];
x-=lowbit(x);
}
return ret;
}
ll Q(int l,int r){
return (r+1ll)*Qsum(t1,r)-l*1ll*Qsum(t1,l-1)-(Qsum(t2,r)-Qsum(t2,l-1));
}
bool check(int x,ll HP){
HP-=Q(1,x)*(1<<cnt);//敌人的初始血量和乘以2的cnt次方
return HP<=0;
}
ll ans;
int main(){
n=read();q=read();W=read();
for(int i=1;i<=n;i++){
a[i]=read();
sum+=a[i];//统计最开始一轮的血量之和
}
for(int i=1;i<=n;i++)add(i,i,a[i]);//建树
while(q--){
ll w=W;//剩余血量
int l=read(),r=read();cnt=0;//cnt表示经过了几轮
ll k=read();
add(l,r,k);//区间加k
sum+=(r-l+1)*k;//敌人的初始血量总和加k
ll now=sum;//敌人现在的血量总和
while(w-now>0){//如果能完整打一轮
ans+=n;//统计
w-=now;//扣血
now*=2;//敌人血量翻倍
cnt++;//统计打了几轮
}//剩下的一轮不完整
int _l=1,_r=n,tmp=0;
while(_l<=_r){//二分确定哪个打出致命攻击
int mid=(_l+_r)>>1;
if(check(mid,w))tmp=mid,_r=mid-1;
else _l=mid+1;
}
ans+=tmp;//加上敌人数
write(ans-1);//致命攻击不算所以-1
ans=0;putchar('\n');
}
return 0;
}