「洛谷1903」「BZOJ2120」「国家集训队」数颜色【带修莫队,树套树】
题目链接
题目大意
单点修改,区间查询有多少种数字。
解法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」「国家集训队」数颜色【带修莫队,树套树】的更多相关文章
- bzoj2120: 数颜色 带修莫队
墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同颜色的画笔. 2. R P ...
- bzoj2120 数颜色——带修莫队
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2120 带修改的莫队: 用结构体存下修改和询问,排好序保证时间后就全局移动修改即可: 参考了T ...
- 【bzoj2120】数颜色 带修莫队
数颜色 题目描述 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到第R支画笔中共有几种不同颜色的画 ...
- BZOJ2120/洛谷P1903 [国家集训队] 数颜色 [带修改莫队]
BZOJ传送门:洛谷传送门 数颜色 题目描述 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会向你发布如下指令: 1. Q L R代表询问你从第L支画笔到第R ...
- BZOJ2120数颜色(带修改莫队)
莫队算法是一种数据结构的根号复杂度替代品,主要应用在询问[l,r]到询问[l+1,r]和[l,r+1]这两个插入和删除操作复杂度都较低的情况下.具体思想是:如果把一个询问[l,r]看做平面上的点(l, ...
- [国家集训队][bzoj2120] 数颜色 [带修改莫队]
题面: 传送门 思路: 这道题和SDOI2009的HH的项链很像,只是多了一个修改 模板套上去呀 莫队学习请戳这里:莫队 Code: #include<iostream> #include ...
- 【BZOJ2120】数颜色(带修莫队)
点此看题面 大致题意:告诉你\(n\)只蜡笔的颜色,有两种操作:第一种操作将第\(x\)只蜡笔颜色改成\(y\),第二种操作询问区间\([l,r]\)内有多少种颜色的蜡笔. 考虑普通莫队 这题目第一眼 ...
- BZOJ2120 数颜色 莫队 带修莫队
原文链接https://www.cnblogs.com/zhouzhendong/p/BZOJ2120.html 题目传送门 - BZOJ2120 题意 给定一个长度为 $n$ 的序列 $a$ ,有 ...
- BZOJ2120 数颜色 【带修莫队】
BZOJ2120 数颜色 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到 ...
随机推荐
- MongoDB——待整理
MongoDB mongoose——http://mongoosejs.com/ npm i mongoose Mongoose 通过外键与另一张表建立关联:Mongoose Populate 基本使 ...
- java kill thread command line
multithreading - How do you kill a Thread in Java? - Stack Overflowhttps://stackoverflow.com/questio ...
- h5-canvas(其他api)
###1.使用图片(需要image对象) drawImage(image,x,y,width,height) 其中image是image或者canvas对象,x和y 是其在目标canvas的起始坐标 ...
- springIOC源码分析(BeanFactroy)
启动spring容器加载bean的方式有两种:最基本的容器BeanFactory和高级容器ApplicationContext.这篇文章介绍使用BeanFactory加载bean时的整个过程,当然,A ...
- v-router几种定义方式
第一种 const router = new VueRouter({ routes: [{ path: '/newSongs', component: require('../views/NewSon ...
- word的"bug"
发表博客发现,从word复制文本到chrome浏览器上的博客时, 如果复制完后立即关闭word,那么将无法粘贴到通过chrome浏览器访问的博客上,也无法粘贴到记事本上: 但是复制完立即关闭word后 ...
- java学习之—排序
package test3; public class Sort{ /** * 冒泡排序 * @param array */ public void bubbleSort(int[] array) { ...
- 浅谈基于Prism的软件系统的架构设计
很早就想写这么一篇文章来对近几年使用Prism框架来设计软件来做一次深入的分析了,但直到最近才开始整理,说到软件系统的设计这里面有太多的学问,只有经过大量的探索才能够设计出好的软件产品,就本人的理解, ...
- 老男孩python学习自修第十二天【常用模块之生成随机数】
常用函数 import random random.random() 生成0到1之间的小数 random.randint(begin, end) 生成[begin, end]之间的整数 random. ...
- SQL之CASE WHEN用法详解[1]
简单CASE WHEN函数: CASE SCORE WHEN 'A' THEN '优' ELSE '不及格' END CASE SCORE WHEN 'B' THEN '良' ELSE '不及格' E ...