题目传送门

sol:第一次看题还真信了是用线段树来做,但是没什么想法,看了题解发现是我不会的Treap,然后花了几天时间学习了一下并补掉题目

  • 无旋Treap

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    const int MAXN = 2e5 + ;
    struct Treap {
    int ls, rs;
    int rand, size;
    LL sum, val, lazy;
    } node[MAXN];
    int root, tot;
    int add_node(int v) {
    int i = ++tot;
    node[i].ls = node[i].rs = ;
    node[i].rand = rand();
    node[i].sum = node[i].val = v;
    node[i].lazy = ;
    node[i].size = ;
    return i;
    }
    void push_down(int rt) {
    int ls = node[rt].ls;
    int rs = node[rt].rs;
    if (ls != ) {
    node[ls].lazy += node[rt].lazy;
    node[ls].val += node[rt].lazy;
    node[ls].sum += node[ls].size * node[rt].lazy;
    }
    if (rs != ) {
    node[rs].lazy += node[rt].lazy;
    node[rs].val += node[rt].lazy;
    node[rs].sum += node[rs].size * node[rt].lazy;
    }
    node[rt].lazy = ;
    }
    void push_up(int rt) {
    int ls = node[rt].ls;
    int rs = node[rt].rs;
    node[rt].size = node[ls].size + node[rs].size + ;
    node[rt].sum = node[ls].sum + node[rs].sum + node[rt].val;
    }
    void split(int rt, int& a, int& b, int s) {
    if (rt == ) {
    a = b = ;
    return;
    }
    push_down(rt);
    int size = node[node[rt].ls].size;
    if (size < s) {
    a = rt;
    split(node[rt].rs, node[a].rs, b, s - size - );
    } else {
    b = rt;
    split(node[rt].ls, a, node[b].ls, s);
    }
    push_up(rt);
    }
    void merge(int& rt, int a, int b) {
    if (a == || b == ) {
    rt = a + b;
    return;
    }
    push_down(a);
    push_down(b);
    if (node[a].rand < node[b].rand) {
    rt = a;
    merge(node[rt].rs, node[a].rs, b);
    } else {
    rt = b;
    merge(node[rt].ls, a, node[b].ls);
    }
    push_up(rt);
    }
    void insert(int p, int v) {
    int x = , y = ;
    split(root, x, y, p - );
    merge(root, x, add_node(v));
    merge(root, root, y);
    }
    void add(int l, int r, int v) {
    int x = , y = , z = ;
    split(root, root, z, r);
    split(root, x, y, l - );
    node[y].lazy += v;
    node[y].val += v;
    node[y].sum += node[y].size * v;
    merge(root, x, y);
    merge(root, root, z);
    }
    LL query(int l, int r) {
    int x = , y = , z = ;
    split(root, root, z, r);
    split(root, x, y, l - );
    LL sum = node[y].sum;
    merge(root, x, y);
    merge(root, root, z);
    return sum;
    }
    int main() {
    int n, q;
    scanf("%d", &n);
    for (int i = ; i <= n; i++) {
    int v; scanf("%d", &v);
    merge(root, root, add_node(v));
    }
    scanf("%d", &q);
    for (int i = ; i <= q; i++) {
    int opt; scanf("%d", &opt);
    if (opt == ) {
    int pos; scanf("%d", &pos);
    insert(pos, );
    } else if (opt == ) {
    int l, r, v;
    scanf("%d%d%d", &l, &r, &v);
    add(l, r, v);
    } else {
    int l, r;
    scanf("%d%d", &l, &r);
    printf("%lld\n", query(l, r));
    }
    }
    return ;
    }

    在前置知识都掌握的情况下Treap还是挺好懂的,而且之前感觉lazy标记挺乱的,看了这题之后感觉理清了

最新文章

  1. Android进程保活
  2. Nginx服务器
  3. Intent官方教程(1)简介和作用
  4. ADO.NET EF实体框架
  5. POJ3126Prime Path
  6. 居然还有WM_TIMECHANGE(只在用户手动改变系统时间时才会产生作用)
  7. Handshakes(思维) 2016(暴力)
  8. Ajax异步请求XMLHttpRequest对象Get请求
  9. C语言之函数的声明
  10. LoadLibrary 失败 GetLastError 126
  11. Python-Django 视图层
  12. JAVA对mongodb的基本操作
  13. sql注入-推断是否存在SQL注入-and大法和or大法
  14. java改单个插入为批量插入
  15. The Little Prince-11/29
  16. 10分钟看懂!基于Zookeeper的分布式锁
  17. SeaJS之use函数
  18. Android Studio - 安卓开发工具 打开后报错集合、修复指南
  19. Django中六个常用的自定义装饰器
  20. Eclipse点不出方法了

热门文章

  1. kubelet组件详解---链接
  2. SpringCloud----服务注册中心Eureka
  3. keras_yolo3阅读
  4. share团队冲刺7
  5. 将元素平分成差值最小的两个集合(DP)
  6. vimdiff换行
  7. 17.3.12--urllib2模块
  8. c# 基础switct---case用于处理多条件的定值的判断
  9. 在MyEclipse的Maven环境下,使用mybatis-generator插件自动生成映射文件(接口)及实体类
  10. latex学习笔记----数学公式