能过样例但是80求调
查看原帖
能过样例但是80求调
846661
ARIS1_0楼主2024/10/23 11:48

树状数组维护区间加区间和

我不会线段树,别让我打线段树

#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;
}
2024/10/23 11:48
加载中...