1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
| #include <bits/stdc++.h> #define endl '\n' using namespace std; using ll = long long; const int N = 5e5 + 10;
int n, m; int a[N];
struct SegTree { int l, r; ll sum, lmax, rmax, max; } t[4 * N];
ll max(ll a, ll b, ll c) { if (b > a) a = b; if (c > a) a = c; return a; }
void pushup(SegTree &p, SegTree &l, SegTree &r) { p.sum = l.sum + r.sum; p.lmax = max(l.lmax, l.sum + r.lmax); p.rmax = max(r.rmax, r.sum + l.rmax); p.max = max(l.max, r.max, l.rmax + r.lmax); }
void build(int p, int l, int r) { if (l == r) { t[p] = {l, l, a[l], a[l], a[l], a[l]}; return; } t[p].l = l, t[p].r = r; int mid = (l + r) / 2; build(p * 2, l, mid); build(p * 2 + 1, mid + 1, r); pushup(t[p], t[p * 2], t[p * 2 + 1]); }
void change(int p, int x, int v) { if (t[p].l == t[p].r) { t[p] = {x, x, v, v, v, v}; return; } int mid = (t[p].l + t[p].r) / 2; if (x <= mid) change(p * 2, x, v); if (x > mid) change(p * 2 + 1, x, v); pushup(t[p], t[p * 2], t[p * 2 + 1]); }
SegTree query(int p, int l, int r) { if (l <= t[p].l && r >= t[p].r) return t[p]; int mid = (t[p].l + t[p].r) / 2; if (r <= mid) return query(p * 2, l, r); if (l > mid) return query(p * 2 + 1, l, r); SegTree a, b, c; a = query(2 * p, l, r); b = query(2 * p + 1, l, r); pushup(c, a, b); return c; }
int main() { ios::sync_with_stdio(false); cin.tie(nullptr); cout.tie(nullptr);
cin >> n >> m; for (int i = 1; i <= n; i++) cin >> a[i]; build(1, 1, n); while (m--) { int op, a, b; cin >> op >> a >> b; if (op == 1) { if (a > b) swap(a, b); cout << query(1, a, b).max << endl; } else change(1, a, b); } }
|