LeetCode 1622 线段树懒标记板子题(c++)

using ll=int;
class SegmentTree{
public:
    const static int N=1e5+1,mod=1e9+7;
    vector<ll> tree,markA,markB;
    SegmentTree(){
        tree.resize(N<<2,0),markA.resize(N<<2,1),markB.resize(N<<2,0);
    }
    void update(int idx, int left, int right, int lo, int hi, int val) {
        if (left >= lo && right <= hi) {
            if (val > 0) {
                markB[idx] = (markB[idx] + val) % mod;
                tree[idx] = (tree[idx] + val) % mod;
            } else {
                markA[idx] = (1LL*markA[idx] * -val) % mod;
                markB[idx] = (1LL*markB[idx] * -val) % mod;
                tree[idx] = (1LL*tree[idx] * -val) % mod;
            }
            return;
        }
        pushDown(idx, left, right);
        int mid = left + ((right - left) >> 1);
        if(lo<=mid) update(idx << 1, left, mid, lo, hi, val);
        if(hi>mid)  update(idx << 1 | 1, mid + 1, right, lo, hi, val);
    }

    int query(int idx, int left, int right, int lo, int hi) {

        if (left >= lo && right <= hi) 
            return tree[idx];
        
        pushDown(idx, left, right);
        int mid = left + ((right - left) >> 1);
        int sum=0;
        if(lo<=mid) sum=query(idx << 1, left, mid, lo, hi);
        if(hi>mid) sum=(sum+query(idx << 1 | 1, mid + 1, right, lo, hi))%mod;
        return sum;
    }

    void pushDown(int idx, int left, int right) {
        if (left <= right && (markA[idx] > 1 || markB[idx] > 0)) {

            markA[idx << 1] = (1LL*markA[idx << 1] * markA[idx]) % mod;
            markB[idx << 1] = (1LL*markB[idx << 1] * markA[idx] + markB[idx]) % mod;
            tree[idx << 1]=(1LL*markA[idx]*tree[idx << 1]+markB[idx])%mod;

            markA[idx << 1 | 1] = (1LL*markA[idx << 1 | 1] * markA[idx]) % mod;
            markB[idx << 1 | 1] = (1LL*markB[idx << 1 | 1] * markA[idx] + markB[idx]) % mod;
            tree[idx << 1|1]=(1LL*markA[idx]*tree[idx << 1|1]+markB[idx])%mod;

            markA[idx] = 1;
            markB[idx] = 0;
        }
    }
};
class Fancy {
public:
    SegmentTree tree;
    int size,n;
    Fancy() {
        size=0,n=1e5;
    }
    void append(int val) {
        tree.update(1, 0, n - 1, size, size, val);
        size++;
    }

    void addAll(int inc) {
        if(size)tree.update(1, 0, n - 1, 0, size - 1, inc);
    }

    void multAll(int m) {
        if(size)tree.update(1, 0, n - 1, 0, size - 1, -m);
    }

    int getIndex(int idx) {
        return idx >= size ? -1 : tree.query(1, 0, n - 1, idx, idx);
    }
};

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值