【每日一题】3月2日区区区间

区区区间

https://ac.nowcoder.com/acm/problem/200195

题意

这个题目线段树三个字都写在题目上了 就没啥好说的了 就是在普通的修改上加上了一个等差数列 因此我们在用lazy维护的时候 可以给lazy打上首项的值 然后其他的就是一个等差数列 在计算的时候也可以通过来判断等差数列的长度 这样知道首项知道等差数列的长度 就可以维护了 然后求和就是用等差数列求和来求即可

code

#include<bits/stdc++.h>
using namespace std;
typedef long long ll; typedef unsigned long long ull; typedef long double ld;
const ll MOD = 1e9 + 7;

const int N = 2e5 + 7;
const int INF = 0x3f3f3f3f;

int n , m ;
int a[N];
ll sum[N << 2] , lazy[N << 2];  

ll calc(ll a1 , ll n){
    return (a1 + a1 + n - 1) * n / 2;
}

void push_up(int rt){
    sum[rt] = sum[rt << 1] + sum[rt << 1 | 1];
}

void push_down(int rt , int l , int r){
    int mid = (l + r) >> 1;
    if(lazy[rt]){
        lazy[rt << 1] = lazy[rt];
        lazy[rt << 1 | 1] = lazy[rt] + (mid - l + 1);
        sum[rt << 1] = calc(lazy[rt << 1] , mid - l + 1);
        sum[rt << 1 | 1] = calc(lazy[rt << 1 | 1] , r - mid);
        lazy[rt] = 0;
    }
}


void update(int rt , int l , int r , int L , int R , int k){
    if(L <= l && r <= R){
        ll a1 = k + l - L , n  = r - l + 1;
        lazy[rt] = a1;
        sum[rt] = calc(a1 , n);
        return ;
    }
    push_down(rt , l , r);
    int mid = (l + r) >> 1 ;
    if(L <= mid) update(rt << 1 , l , mid , L , R , k);
    if(R > mid) update(rt << 1 | 1 , mid + 1, r , L , R , k);
    push_up(rt);
}

ll query(int rt , int l , int r , int L , int R){
    if(L <= l && r <= R )   return sum[rt];
    push_down(rt , l , r);
    ll res = 0;
    int mid = (l + r) >> 1;
    if(L <= mid)    res += query(rt << 1, l , mid , L , R);
    if(R > mid)     res += query(rt << 1 | 1 ,  mid + 1 , r , L ,R);
    return res;
}
void build(int rt , int l , int r){
    if(l == r){
        sum[rt] = a[l];
        return ;
    }
    int mid = (l + r) >> 1;
    build(rt << 1 , l , mid);
    build(rt << 1 | 1 , mid + 1 , r);
    push_up(rt);
}

int main(void){
    scanf("%d%d" , &n , &m);
    for(int i = 1 ; i <= n  ; ++i)  scanf("%d" , &a[i]);   
    build(1 , 1 , n);
    while(m--){
        int op , l , r;
        scanf("%d%d%d" , &op , &l , &r);
        if(op & 1){
            int k;
            scanf("%d" , &k);
            update(1 , 1 , n , l , r , k);
        }
        else{
            ll ans = query(1 , 1 , n , l , r);
            printf("%lld\n" , ans);
        }
    }
    return 0;
}
每日一题 文章被收录于专栏

写每日一题呀

全部评论

相关推荐

Yushuu:你的确很厉害,但是有一个小问题:谁问你了?我的意思是,谁在意?我告诉你,根本没人问你,在我们之中0人问了你,我把所有问你的人都请来 party 了,到场人数是0个人,誰问你了?WHO ASKED?谁问汝矣?誰があなたに聞きましたか?누가 물어봤어?我爬上了珠穆朗玛峰也没找到谁问你了,我刚刚潜入了世界上最大的射电望远镜也没开到那个问你的人的盒,在找到谁问你之前我连癌症的解药都发明了出来,我开了最大距离渲染也没找到谁问你了我活在这个被辐射蹂躏了多年的破碎世界的坟墓里目睹全球核战争把人类文明毁灭也没见到谁问你了😆
点赞 评论 收藏
分享
求个公司要我:接好运
点赞 评论 收藏
分享
评论
1
收藏
分享
牛客网
牛客企业服务