結果

問題 No.3507 RangeSum RangeUpdate RangeSqrt
コンテスト
ユーザー Enderaoe Lyther
提出日時 2026-04-17 21:43:20
言語 C++23
(gcc 15.2.0 + boost 1.89.0)
コンパイル:
g++-15 -O2 -lm -std=c++23 -Wuninitialized -DONLINE_JUDGE -o a.out _filename_
実行:
./a.out
結果
AC  
実行時間 305 ms / 2,000 ms
コード長 4,769 bytes
記録
記録タグの例:
初AC ショートコード 純ショートコード 純主流ショートコード 最速実行時間
コンパイル時間 2,366 ms
コンパイル使用メモリ 189,952 KB
実行使用メモリ 16,256 KB
最終ジャッジ日時 2026-04-17 21:43:29
合計ジャッジ時間 9,393 ms
ジャッジサーバーID
(参考情報)
judge3_1 / judge2_0
このコードへのチャレンジ
(要ログイン)
ファイルパターン 結果
sample AC * 1
other AC * 29
権限があれば一括ダウンロードができます

ソースコード

diff #
raw source code

#include <algorithm>
#include <cmath>
#include <iostream>
#include <vector>

using namespace std;

struct SegTree {
    struct Node {
        long long sum = 0;
        int mn = 0;
        int mx = 0;
        long long add = 0;
        bool has_assign = false;
        int assign = 0;
    };

    int n;
    vector<Node> seg;

    explicit SegTree(const vector<int>& a) : n((int)a.size()), seg(4 * n) {
        build(1, 0, n, a);
    }

    static int isqrt_int(int x) {
        int r = (int)std::sqrt((long double)x);
        while (1LL * (r + 1) * (r + 1) <= x) ++r;
        while (1LL * r * r > x) --r;
        return r;
    }

    void build(int idx, int l, int r, const vector<int>& a) {
        if (r - l == 1) {
            seg[idx].sum = a[l];
            seg[idx].mn = a[l];
            seg[idx].mx = a[l];
            return;
        }
        int mid = (l + r) >> 1;
        build(idx << 1, l, mid, a);
        build(idx << 1 | 1, mid, r, a);
        pull(idx);
    }

    void pull(int idx) {
        seg[idx].sum = seg[idx << 1].sum + seg[idx << 1 | 1].sum;
        seg[idx].mn = min(seg[idx << 1].mn, seg[idx << 1 | 1].mn);
        seg[idx].mx = max(seg[idx << 1].mx, seg[idx << 1 | 1].mx);
    }

    void apply_assign(int idx, int l, int r, int v) {
        seg[idx].sum = 1LL * (r - l) * v;
        seg[idx].mn = v;
        seg[idx].mx = v;
        seg[idx].add = 0;
        seg[idx].has_assign = true;
        seg[idx].assign = v;
    }

    void apply_add(int idx, int l, int r, int delta) {
        if (delta == 0) return;
        seg[idx].sum += 1LL * (r - l) * delta;
        seg[idx].mn += delta;
        seg[idx].mx += delta;
        if (seg[idx].has_assign) {
            seg[idx].assign += delta;
        } else {
            seg[idx].add += delta;
        }
    }

    void push(int idx, int l, int r) {
        if (r - l == 1) {
            seg[idx].add = 0;
            seg[idx].has_assign = false;
            return;
        }

        int mid = (l + r) >> 1;
        if (seg[idx].has_assign) {
            apply_assign(idx << 1, l, mid, seg[idx].assign);
            apply_assign(idx << 1 | 1, mid, r, seg[idx].assign);
            seg[idx].has_assign = false;
        }
        if (seg[idx].add != 0) {
            apply_add(idx << 1, l, mid, (int)seg[idx].add);
            apply_add(idx << 1 | 1, mid, r, (int)seg[idx].add);
            seg[idx].add = 0;
        }
    }

    void range_assign(int idx, int l, int r, int ql, int qr, int x) {
        if (qr <= l || r <= ql) return;
        if (ql <= l && r <= qr) {
            apply_assign(idx, l, r, x);
            return;
        }
        push(idx, l, r);
        int mid = (l + r) >> 1;
        range_assign(idx << 1, l, mid, ql, qr, x);
        range_assign(idx << 1 | 1, mid, r, ql, qr, x);
        pull(idx);
    }

    void range_sqrt(int idx, int l, int r, int ql, int qr) {
        if (qr <= l || r <= ql || seg[idx].mx <= 1) return;

        if (ql <= l && r <= qr) {
            int sa = isqrt_int(seg[idx].mn);
            int sb = isqrt_int(seg[idx].mx);
            if (sa == sb) {
                apply_assign(idx, l, r, sa);
                return;
            }
            if (sa - seg[idx].mn == sb - seg[idx].mx) {
                apply_add(idx, l, r, sa - seg[idx].mn);
                return;
            }
        }

        if (r - l == 1) {
            apply_assign(idx, l, r, isqrt_int(seg[idx].mx));
            return;
        }

        push(idx, l, r);
        int mid = (l + r) >> 1;
        range_sqrt(idx << 1, l, mid, ql, qr);
        range_sqrt(idx << 1 | 1, mid, r, ql, qr);
        pull(idx);
    }

    long long range_sum(int idx, int l, int r, int ql, int qr) {
        if (qr <= l || r <= ql) return 0;
        if (ql <= l && r <= qr) return seg[idx].sum;
        push(idx, l, r);
        int mid = (l + r) >> 1;
        return range_sum(idx << 1, l, mid, ql, qr)
             + range_sum(idx << 1 | 1, mid, r, ql, qr);
    }

    void range_assign(int l, int r, int x) {
        range_assign(1, 0, n, l, r, x);
    }

    void range_sqrt(int l, int r) {
        range_sqrt(1, 0, n, l, r);
    }

    long long range_sum(int l, int r) {
        return range_sum(1, 0, n, l, r);
    }
};

int main() {
    ios::sync_with_stdio(false);
    cin.tie(nullptr);

    int N, Q;
    cin >> N >> Q;

    vector<int> A(N);
    for (int i = 0; i < N; ++i) cin >> A[i];

    SegTree seg(A);

    for (int qi = 0; qi < Q; ++qi) {
        int type, l, r;
        cin >> type >> l >> r;
        if (type == 0) {
            cout << seg.range_sum(l, r) << '\n';
        } else if (type == 1) {
            int x;
            cin >> x;
            seg.range_assign(l, r, x);
        } else {
            seg.range_sqrt(l, r);
        }
    }
    return 0;
}
0