题目链接

【BZOJ传送门】
【洛谷传送门】

题目大意

单点修改,区间查询有多少种数字。

解法1--树套树

可以直接暴力树套树,我比较懒,不想写。
稍微口胡一下,可以直接来一个树状数组套主席树,也就是待修的主席树。
然后查询的时候,两个根节点减一下就可以了。

解法2--带修莫队

这是带修莫队的模板题。
最简单的莫队是是一个二元组\((l,r)\),这里引入了一个新的参数,变成了三元组\((l,r,t)\),\(t\)所表示的是在这个查询最前面的哪一个修改的编号。
然后我们这个\(t\)当做第三关键字,进行询问的排序。

bool cmp(const Que_rec &A, const Que_rec &B) { return A.l / block == B.l / block ? (A.r / block == B.r / block ? A.t < B.t : A.r < B.r) : A.l < B.l; }

也就是把\(r\)也进行块的排序。
那么在查询的最后,我们需要加上修改操作。
因为如果一个东西修改过后,但是我们先查询了,那么就把这个东西变回去,然后再修改就好了。

代码

开了\(O(2)\)才过掉的垃圾代码。

#include <bits/stdc++.h>
#pragma GCC optimize
#define N 50105
#define M 1000005
using namespace std;
template <typename T> void read(T &x) {
    x = 0; T fl = 1; char ch = 0;
    for (; ch < '0' || ch > '9'; ch = getchar()) if (ch == '-') fl = -1;
    for (; ch >= '0' && ch <= '9'; ch = getchar()) x = (x << 1) + (x << 3) + (ch ^ 48);
    x *= fl;
}
struct Que_rec { int l, r, t, id; } q[N];
struct Moy_rec { int id, val; } mo[N];
int block, n, m, q_cnt = 0, m_cnt = 0, res;
int a[N], ddd[M], ans[N];
bool cmp(const Que_rec &A, const Que_rec &B) { return A.l / block == B.l / block ? (A.r / block == B.r / block ? A.t < B.t : A.r < B.r) : A.l < B.l; }
void upda(int x) { if (ddd[x] ++ == 0) res ++; }
void updd(int x) { if (-- ddd[x] == 0) res --; }
void modify(int lmo, int id) {
    if (mo[lmo].id >= q[id].l && mo[lmo].id <= q[id].r) updd(a[mo[lmo].id]), upda(mo[lmo].val);
    swap(a[mo[lmo].id], mo[lmo].val);
}
int main() {
    read(n); read(m); block = (int)sqrt(1.0 * n + 0.5) * 2;
    for (int i = 1; i <= n; i ++) read(a[i]);
    for (int i = 1; i <= m; i ++) {
        char opt[5]; scanf("%s", opt); int x, y;
        if (opt[0] == 'Q') { read(x); read(y); ++ q_cnt; q[q_cnt] = (Que_rec){x, y, m_cnt, q_cnt}; }
        else { m_cnt ++; read(x); read(y); mo[m_cnt] = (Moy_rec){x, y}; }
    }
    sort(q + 1, q + 1 + q_cnt, cmp);
    int l = 2, r = 1, lmo = 0; res = 0;
    for (int i = 1; i <= q_cnt; i ++) {
        while (r < q[i].r) upda(a[++ r]);
        while (r > q[i].r) updd(a[r --]);
        while (l > q[i].l) upda(a[-- l]);
        while (l < q[i].l) updd(a[l ++]);
        while (lmo < q[i].t) modify(++ lmo, i);
        while (lmo > q[i].t) modify(lmo --, i);
        ans[q[i].id] = res;
    }
    for (int i = 1; i <= q_cnt; i ++) printf("%d\n", ans[i]);
    return 0;
}

「洛谷1903」「BZOJ2120」「国家集训队」数颜色【带修莫队,树套树】的更多相关文章

  1. bzoj2120: 数颜色 带修莫队

    墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同颜色的画笔. 2. R P ...

  2. bzoj2120 数颜色——带修莫队

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2120 带修改的莫队: 用结构体存下修改和询问,排好序保证时间后就全局移动修改即可: 参考了T ...

  3. 【bzoj2120】数颜色 带修莫队

    数颜色 题目描述 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同颜色的画 ...

  4. BZOJ2120/洛谷P1903 [国家集训队] 数颜色 [带修改莫队]

    BZOJ传送门:洛谷传送门 数颜色 题目描述 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会向你发布如下指令: 1. Q L R代表询问你从第L支画笔到第R ...

  5. BZOJ2120数颜色(带修改莫队)

    莫队算法是一种数据结构的根号复杂度替代品,主要应用在询问[l,r]到询问[l+1,r]和[l,r+1]这两个插入和删除操作复杂度都较低的情况下.具体思想是:如果把一个询问[l,r]看做平面上的点(l, ...

  6. [国家集训队][bzoj2120] 数颜色 [带修改莫队]

    题面: 传送门 思路: 这道题和SDOI2009的HH的项链很像,只是多了一个修改 模板套上去呀 莫队学习请戳这里:莫队 Code: #include<iostream> #include ...

  7. 【BZOJ2120】数颜色(带修莫队)

    点此看题面 大致题意:告诉你\(n\)只蜡笔的颜色,有两种操作:第一种操作将第\(x\)只蜡笔颜色改成\(y\),第二种操作询问区间\([l,r]\)内有多少种颜色的蜡笔. 考虑普通莫队 这题目第一眼 ...

  8. BZOJ2120 数颜色 莫队 带修莫队

    原文链接https://www.cnblogs.com/zhouzhendong/p/BZOJ2120.html 题目传送门 - BZOJ2120 题意 给定一个长度为 $n$ 的序列 $a$ ,有 ...

  9. BZOJ2120 数颜色 【带修莫队】

    BZOJ2120 数颜色 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到 ...

随机推荐

  1. ad2017安装以及破解

    1.破Ad破解https://wenku.baidu.com/view/5e23a78e2e3f5727a5e962dd.html 2. Ad 汉化https://jingyan.baidu.com/ ...

  2. Shell脚本2

      5 Shell传递参数 我们可以在执行 Shell 脚本时,向脚本传递参数, 脚本内获取参数的格式为:$n.n 代表一个数字,1 为执行脚本的第一个参数,2 为执行脚本的第二个参数,以此类推…… ...

  3. Failure to transfer org.apache.maven:maven-archiver:pom:2.5 from https://repo.maven.apache.org/maven2 was cached in the local repository, resolution will not be reattempted until the update interval o

    pom.xml报错: Failure to transfer org.apache.maven:maven-archiver:pom:2.5 from https://repo.maven.apach ...

  4. WCF上传下载文件

    思路:上传时将要上传的文件流提交给服务器端 下载时只需要将服务器上的流返回给客户端即可 1.契约,当需要传递的数量多于一个时就需要通过messagecontract来封装起来 这里分别实现了上传和下载 ...

  5. 【Python3练习题 020】 求1+2!+3!+...+20!的和

    方法一 import functools   sum = 0 for i in range(1,21):     sum = sum + functools.reduce(lambda x,y: x* ...

  6. python3 Tkinter GUI 试水

    from tkinter import * #导入tkinter下所有包,用于GUI开发#窗口创建tk=Tk()cans=Canvas(tk,width=400,height=400)#定义窗口规格c ...

  7. 【学亮IT手记】jQuery each()函数用法实例

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <script sr ...

  8. VSC软件快捷键

    Shift + Alt + F  格式化 Ctrl+Shift+P, F1显示命令面板 Ctrl+P快速打开,进入File… Ctrl + Shift + N新窗口/实例 Ctrl + Shift + ...

  9. linux audit (9)--生成audit报表

    aureport这个命令可以生成一个总结性的柱状图报表,默认情况下,在/var/log/audit目录下的所有日志文件都会生成一个报表,也可以使用如下命令来指定一个不同的文件,aureport opt ...

  10. java 中的打印流

    package cn.zhou; import java.io.BufferedInputStream; import java.io.BufferedReader; import java.io.F ...