P3168 [CQOI2015]任务查询系统

时间:2021-11-13 06:04:25

题目地址:P3168 [CQOI2015]任务查询系统

主席树的模板题

更模板的在这儿:P3834 【模板】可持久化线段树 1(主席树)

形象的说,P3834是“单点修改,区间查询”,P3168是“区间修改,单点查询”

注意!这里只是形象的说,实际上两道题都是静态的

我们联想其他的“区间修改,单点查询”问题,我们都是怎么做的?

差分!

没错,差分,将“区间修改”改成“左端点加,右端点的右边减”的“单点修改”

那么,我们每次“单点查询”就相当于求前缀和,也就是一个“区间查询”

于是我们成功的将P3168转化成了P3834,然后就简单了

几个注意点:

  • 公式 \(K_i=1+(A_i*Pre+B_i)\ mod\ C_i\) 会爆 int,注意开 long long

  • 需要先进行离散化, \(sort+unique\) 即可

  • 主席树要开 \(2*N\ log\ N\) ,其中 \(2\) 是因为差分以后“单点修改”的数量为原来“区间修改”的两倍(我的代码中直接将 N 扩大了一倍)

  • 不要修改第 \(n+1\) 个区间,即修改到第 \(n+1\) 个区间就可以直接 break

  • 在询问到 \(l==r\) 时不能直接返回 \(sum\) 而要返回 \(t[p].s / t[p].c * k\)

#include <bits/stdc++.h>
#define ll long long
#define pii pair<int, int>
#define x first
#define y second
using namespace std;
const int N = 2e5 + 6;
int n, m, pre = 1, b[N], rt[N], tot;
pii a[N];
struct T {
    int l, r, c, s;
} t[N*20];

int build(int l, int r) {
    int p = ++tot;
    if (l == r) return p;
    int mid = (l + r) >> 1;
    t[p].l = build(l, mid);
    t[p].r = build(mid + 1, r);
    return p;
}

int change(int o, int l, int r, int p, int k) {
    int q = ++tot;
    t[q] = t[o];
    if (l == r) {
        t[q].c += k;
        t[q].s += k * b[p];
        return q;
    }
    int mid = (l + r) >> 1;
    if (p <= mid) t[q].l = change(t[o].l, l, mid, p, k);
    else t[q].r = change(t[o].r, mid + 1, r, p, k);
    t[q].c = t[t[q].l].c + t[t[q].r].c;
    t[q].s = t[t[q].l].s + t[t[q].r].s;
    return q;
}

int ask(int p, int l, int r, int k) {
    if (l == r) return t[p].s / t[p].c * k;
    int mid = (l + r) >> 1;
    if (k <= t[t[p].l].c) return ask(t[p].l, l, mid, k);
    else return t[t[p].l].s + ask(t[p].r, mid + 1, r, k - t[t[p].l].c);
}

int main() {
    cin >> n >> m;
    for (int i = 1; i <= n; i++) {
        int p = (i << 1) - 1, q = i << 1;
        scanf("%d %d %d", &a[p].x, &a[q].x, &a[p].y);
        ++a[q].x;
        a[q].y = -a[p].y;
        b[i] = a[p].y;
    }
    sort(b + 1, b + n + 1);
    int w = unique(b + 1, b + n + 1) - (b + 1);
    rt[0] = build(1, w);
    sort(a + 1, a + (n << 1) + 1);
    int k = 1;
    for (int i = 1; i <= (n << 1); i++) {
        while (k < a[i].x) {
            rt[k+1] = rt[k];
            ++k;
        }
        if (k == n + 1) break;
        int p = lower_bound(b + 1, b + w + 1, abs(a[i].y)) - b;
        rt[k] = change(rt[k], 1, w, p, a[i].y > 0 ? 1 : -1);
    }
    while (m--) {
        int xi, ai, bi, ci;
        scanf("%d %d %d %d", &xi, &ai, &bi, &ci);
        int ki = ((ll)ai * pre + bi) % ci + 1;
        if (t[rt[xi]].c <= ki) printf("%d\n", pre = t[rt[xi]].s);
        else printf("%d\n", pre = ask(rt[xi], 1, w, ki));
    }
    return 0;
}

P3168 [CQOI2015]任务查询系统的更多相关文章

  1. bzoj3932 &sol; P3168 &lbrack;CQOI2015&rsqb;任务查询系统(主席树&plus;差分)

    P3168 [CQOI2015]任务查询系统 看到第k小,就是主席树辣 对于每一段任务(a,b,k),在版本a的主席树+k,版本b+1的主席树-k 同一时间可能有多次修改,所以开个vector存操作, ...

  2. 洛谷 P3168 &lbrack;CQOI2015&rsqb;任务查询系统 解题报告

    P3168 [CQOI2015]任务查询系统 题目描述 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分. 超级计算机中的任务用三元组\((S_i,E_i,P_i) ...

  3. 洛谷P3168 &lbrack;CQOI2015&rsqb;任务查询系统 &lbrack;主席树,差分&rsqb;

    题目传送门 任务查询系统 题目描述 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任 ...

  4. ●洛谷P3168 &lbrack;CQOI2015&rsqb;任务查询系统

    题链: https://www.luogu.org/problemnew/show/P3168题解: 主席树 强制在线? 那就直接对每一个前缀时间建一个线段树(可持久化线段树),线段树维护优先度权值. ...

  5. P3168 &lbrack;CQOI2015&rsqb;任务查询系统&lpar;主席树&rpar;

    题目描述 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第Ei ...

  6. 洛谷P3168 &lbrack;CQOI2015&rsqb;任务查询系统

    #include<cstdio> #include<cstdlib> #include<algorithm> #include<cstring> #in ...

  7. Luogu P3168 &lbrack;CQOI2015&rsqb;任务查询系统

    题目链接 \(Click\) \(Here\) 差分主席树,就是把主席树做成一个差分前缀和的形式,还是很容易想到的. 写主席树的时候几个注意点: 查询可能开始于所有任务之前,二分任务点要把左边界设置为 ...

  8. p3168 &lbrack;CQOI2015&rsqb;任务查询系统(差分&plus;主席树)

    恕我才学浅薄,一开始想到的是树状数组+线段树,然后看了题解才第一次见到了差分这种神奇的科技 仔细想想,主席树的本质不就是前缀和嘛,加上一个差分也是可以的,没想到真是罪过罪过 对时间维护一个差分 在Si ...

  9. 「Luogu P3168 &lbrack;CQOI2015&rsqb;任务查询系统」

    介绍本题的两种做法: 方法1 前置芝士 线段树:一个很重要的数据结构. 树状数组:一个很重要的数据结构. 具体实现 区间修改,单点查询很容易就会想到树状数组了,至于查询前k个数的和又可以丢给权值线段树 ...

随机推荐

  1. java 的 sqlHelper,改改之后也适用于不使用 EF 的 C&num; 项目,包含查询和建表。

    这个类用来拼接 sql. package com.ly.orm; public class Query { protected Query(String v) { sql = v; } public ...

  2. sql server2008 字段类型

    bit    整型 bit数据类型是整型,其值只能是0.1或空值.这种数据类型用于存储只有两种可能值的数据,如Yes 或No.True 或False .On 或Off. 注意:很省空间的一种数据类型, ...

  3. 软件工程课后作业——用JAVA编写的随机产生30道四则运算

    package com.java.sizeyunsuan; public class lianxi { String f() { int i=(int)(Math.random()*10); int ...

  4. spl&lowbar;autoload&lowbar;register array参数

    spl_autoload_register (PHP 5 >= 5.1.2) spl_autoload_register — 注册给定的函数作为 __autoload 的实现 说明¶ bool  ...

  5. Fixing common issues when hosting a &period;NET 4&period;0 WCF service in IIS 7

    http://sandrinodimattia.net/fixing-common-issues-when-hosting-a-net-4-0-wcf-service-in-iis-7/ Until ...

  6. poj 2375 Cow Ski Area bfs

    这个题目用tarjan找联通块,缩点,然后统计出入度为0的点理论上是可行的,但问题是会暴栈.考虑到这个题目的特殊性,可以直接用一次bfs找到数字相同且联通的块,这就是一个联通块,然后缩点,统计出入度即 ...

  7. JavaScript事件循环&lpar;Event Loop&rpar;机制

    JavaScript 是单线程单并发语言 什么是单线程 主程序只有一个线程,即同一时间片断内其只能执行单个任务. 为什么选择单线程? JavaScript的主要用途是与用户互动,以及操作DOM.这决定 ...

  8. 64位windows8&period;1下安装 ImageMagick 总结

    1.  安装 ImageMagick-6.7.7-Q16-x64 下载地址:http://ftp.sunet.se/pub/multimedia/graphics/ImageMagick/binari ...

  9. Docker最全教程——从理论到实战&lpar;一&rpar;

    容器是应用走向云端之后必然的发展趋势,因此笔者非常乐于和大家分享我们这段时间对容器的理解.心得和实践. 本篇教程持续编写了2个星期左右,只是为了大家更好地了解.理解和消化这个技术,能够搭上这波车. 你 ...

  10. left join inner join 区别

    left 以左表为准,左表在右表没有对应的记录,也为显示(右表字段填空). inner 需要满足两张表都有记录. 不管哪种join 一对多最终的结局 只会是多条记录