版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/qq_21110267/article/details/44889427
描述
区间修改, 区间求和
分析
- 想练练splay打标记.
- 因为splay不支持永久标号, 所以pushdown后必须把标记清掉.
- 第一次打上标记后要立刻让标记生效.
- 需要注意的地方是pushdown必须让子结点的标记生效.
#include
using namespace std;
typedef long long int lli;
const int maxn = 200000 + 10;
int s[maxn], ch[maxn][2];
lli v[maxn], sumv[maxn], inc[maxn];
#define lc ch[o][0]
#define rc ch[o][1]
#define seq ch[rc][0]
void update(int o) {
s[o] = s[lc] + s[rc] + 1;
sumv[o] = sumv[lc] + sumv[rc] + v[o];
}
void pushdown(int o) {
if(!inc[o]) return;
inc[lc] += inc[o]; v[lc] += inc[o]; sumv[lc] += inc[o] * s[lc];
inc[rc] += inc[o]; v[rc] += inc[o]; sumv[rc] += inc[o] * s[rc];
inc[o] = 0;
}
int cmp(int o, int k) {
if(s[lc] + 1 == k) return -1;
return k < s[lc] + 1 ? 0 : 1;
}
void rotate(int& o, int d) {
int p = ch[o][d^1]; ch[o][d^1] = ch[p][d]; ch[p][d] = o;
update(o); update(p); o = p;
}
void splay(int& o, int k) {
pushdown(o);
int d = cmp(o, k);
if(d == -1) return;
if(d == 1) k -= (s[lc] + 1);
int p = ch[o][d];
pushdown(p);
int d2 = cmp(p, k);
if(d2 != -1) {
int k2 = (d2 == 0 ? k : k-s[ch[p][0]]-1);
splay(ch[p][d2], k2);
if(d == d2) rotate(o, d^1); else rotate(ch[o][d], d);
}
rotate(o, d^1);
}
void build(int& o, int L, int R) {
if(L > R) return;
o = (L+R)>>1;
build(lc, L, o-1); build(rc, o+1, R);
update(o);
}
int main() {
int n, m, o;
scanf("%d", &n); n += 2;
for(int i = 2; i < n; i++) scanf("%d", &v[i]);
build(o, 1, n);
scanf("%d", &m);
for(int i = 0; i < m; i++) {
lli x;
int k, y1, y2;
scanf("%d %d %d", &k, &y1, &y2);
y1++; y2++;
splay(o, y1-1);
splay(rc, y2+1-(s[lc]+1));
if(k == 2) printf("%lld\n", sumv[seq]);
else {
scanf("%lld", &x);
inc[seq] += x; v[seq] += x; sumv[seq] += s[seq] * x;
update(rc); update(o);
}
}
return 0;
}
#include
using namespace std;
typedef long long lli;
const int maxn = 600000;
lli y1, y2, v;
lli addv[maxn];
lli sumv[maxn];
#define lc (o<<1)
#define rc (o<<1^1)
#define M (L+R>>1)
void build(int o, int L, int R) {
if(L == R) scanf("%lld", &sumv[o]);
else {
build(lc, L, M);
build(rc, M+1, R);
sumv[o] = sumv[lc] + sumv[rc];
}
}
void update(int o, int L, int R) {
if(y1 <= L && y2 >= R) {
addv[o] += v;
sumv[o] += v*(R-L+1);
} else {
if(y1 <= M) update(lc, L, M);
if(y2 > M) update(rc, M+1, R);
sumv[o] = sumv[lc] + sumv[rc] + addv[o] * (R-L+1);
}
}
lli sum_v;
void query(int o, int L, int R, lli add) {
if(y1 <= L && R <= y2) sum_v += sumv[o] + add*(R-L+1);
else {
if(y1 <= M) query(lc, L, M, add + addv[o]);
if(y2 > M) query(rc, M+1, R, add + addv[o]);
}
}
int main() {
int n, m;
scanf("%d", &n);
build(1, 1, n);
scanf("%d", &m);
for(int i = 1; i <= m; i++) {
int k;
scanf("%d", &k);
if(k == 1) {
scanf("%lld%lld%lld", &y1, &y2, &v);
update(1, 1, n);
} else {
scanf("%lld%lld", &y1, &y2);
sum_v = 0;
query(1, 1, n, 0);
printf("%lld\n", sum_v);
}
}
return 0;
}
#include
using namespace std;
typedef long long lli;
const int maxn = 200000 + 10;
lli n, c1[maxn], c2[maxn], s[maxn];
lli query(int x, lli* c) {
lli ret = 0;
for(; x > 0; x += x&-x) ret += c[x];
return ret;
}
void modify(int x, int d, lli* c) {
for(; x <= n; x += x&-x) c[x] += d;
}
int main() {
scanf("%d", &n);
for(int i = 1; i <= n; i++) {
scanf("%lld", &s[i]);
s[i] += s[i-1];
}
int m, k, d, L, R;
scanf("%d", &m);
for(int i = 1; i <= m; i++) {
scanf("%d", &k);
if(k == 1) {
scanf("%d%d%d", &L, &R, &d);
modify(L, d, c1);
modify(R+1, -d, c1);
modify(L, L*d, c2);
modify(R+1, -(R+1)*d, c2);
} else {
scanf("%d%d", &L, &R);
lli s1 = s[R]-s[L-1];
lli s2 = query(R,c1)*(R+1) - query(L-1,c1)*L;
lli s3 = query(L-1,c2) - query(R,c2);
printf("%lld\n", s1 + s2 + s3);
}
}
return 0;
}