【bzoj2453】维护队列/【bzoj2120】数颜色 分块+二分
题目描述
输入
输出
样例输入
2 3
1 2
Q 1 2
R 1 2
Q 1 2
样例输出
2
1
题解
分块+二分
两道一样的题。。。
对于每个i记录一个p[i],代表i前一个颜色与i相同的位置,没有则为-1。
那么对于区间[l,r]中的i,如果p[i]<l,那么i就是[l,r]中第一个出现c[i]颜色的,可以记录到答案中。
然后分成√n 个块,建立新数组对块中p[i]排序。
查询时,整块使用排序后的数组二分查找小于l的个数,多余部分暴力查询。
修改时,需要考虑3部分:i后面第一个c[j]==c[x],i后面第一个c[j]==y,i前面第一个c[j]==y。
于是我们还需要建立新数组对块中c[i]排序,并使用二分查找来完成这三个操作。
具体方法:x块内暴力查找,x块外二分查找。
这里压了大量的行,并使用flag减小花括号,不然太多行实在要看吐了QAQ。
还是别忘特判,另外注意i的含义,位置号或块号。
看到网上还有暴力修改的方法,不过亲测很慢,不推荐。
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
using namespace std;
int n , si , c[10010] , vc[10010] , p[10010] , vp[10010] , last[1000010];
char str[5];
void reset(int b)
{
int l = b * si , r = min((b + 1) * si - 1 , n - 1) , i;
for(i = l ; i <= r ; i ++ ) vc[i] = c[i] , vp[i] = p[i];
sort(vc + l , vc + r + 1) , sort(vp + l , vp + r + 1);
}
int query(int b , int lim)
{
int l = b * si , r = min((b + 1) * si - 1 , n - 1) , mid , ans = l - 1 , tl = l;
while(l <= r)
{
mid = (l + r) >> 1;
if(vp[mid] < lim) ans = mid , l = mid + 1;
else r = mid - 1;
}
return ans - tl + 1;
}
bool findc(int l , int r , int x)
{
int mid;
while(l <= r)
{
mid = (l + r) >> 1;
if(vc[mid] == x) return 1;
else if(vc[mid] < x) l = mid + 1;
else r = mid - 1;
}
return 0;
}
int main()
{
int m , i , j , x , y , ans , flag , l , r;
scanf("%d%d" , &n , &m);
si = (int)sqrt(n);
memset(last , -1 , sizeof(last));
for(i = 0 ; i < n ; i ++ ) scanf("%d" , &c[i]) , p[i] = last[c[i]] , last[c[i]] = i;
for(i = 0 ; i * si < n ; i ++ ) reset(i);
while(m -- )
{
scanf("%s%d%d" , str , &x , &y);
x -- ;
if(str[0] == 'Q')
{
y -- ;
ans = 0;
if(x / si == y / si) for(i = x ; i <= y ; i ++ ) ans += (p[i] < x);
else
{
for(i = x / si + 1 ; i < y / si ; i ++ ) ans += query(i , x);
for(i = x ; i < (x / si + 1) * si ; i ++ ) ans += (p[i] < x);
for(i = y / si * si ; i <= y ; i ++ ) ans += (p[i] < x);
}
printf("%d\n" , ans);
}
else
{
if(c[x] == y) continue;
flag = 0;
for(i = x + 1 ; !flag && i < (x / si + 1) * si && i < n ; i ++ ) if(p[i] == x) p[i] = p[x] , reset(i / si) , flag = 1;
for(i = x / si + 1 ; !flag && i * si < n ; i ++ )
{
l = i * si , r = min((i + 1) * si - 1 , n - 1);
if(findc(l , r , c[x])) for(j = l ; !flag && j <= r ; j ++ ) if(c[j] == c[x]) p[j] = p[x] , reset(i) , flag = 1;
}
c[x] = y;
flag = 0;
for(i = x + 1 ; !flag && i < (x / si + 1) * si && i < n ; i ++ ) if(c[i] == c[x]) p[i] = x , reset(i / si) , flag = 1;
for(i = x / si + 1 ; !flag && i * si < n ; i ++ )
{
l = i * si , r = min((i + 1) * si - 1 , n - 1);
if(findc(l , r , c[x])) for(j = l ; !flag && j <= r ; j ++ ) if(c[j] == c[x]) p[j] = x , reset(i) , flag = 1;
}
flag = 0;
for(i = x - 1 ; !flag && i >= x / si * si ; i -- ) if(c[i] == c[x]) p[x] = i , flag = 1;
for(i = x / si - 1 ; !flag && ~i ; i -- )
{
l = i * si , r = min((i + 1) * si - 1 , n - 1);
if(findc(l , r , c[x])) for(j = r ; !flag && j >= l ; j -- ) if(c[j] == c[x]) p[x] = j , flag = 1;
}
if(!flag) p[x] = -1;
reset(x / si);
}
}
}
【bzoj2453】维护队列/【bzoj2120】数颜色 分块+二分的更多相关文章
- BZOJ2453维护队列&&BZOJ2120数颜色
2016-05-28 11:20:22 共同的思路: 维护某种颜色上一次在哪里出现pre,可以知道当pre<询问的l时更新答案 块内按照pre排序 修改的时候重新O(n)扫一遍,如果和之前的不一 ...
- [BZOJ2120] 数颜色 && [bzoj2453] 维护队列(莫队 || 分块)
传送门 只有第一个,第二个权限题. 分块,然而wa,没看出来错在哪里,有时间再看. #include <cmath> #include <cstdio> #include &l ...
- bzoj2120 数颜色 分块
分块大法好 orz 处理出每个点的前驱和后继位置. 暴力修改,查询就在每个整块里查询pre<l的,暴力跑两边就好了 #include<cstdio> #include<cstr ...
- Luogu 1903 数颜色 | 分块
Luogu 1903 数颜色 | 分块 莫队不会啊-- 这道题直接分块也能卡过! 这道题的做法很有趣:对于每个位置i,记录它的颜色a[i]上一次出现的位置,记为pre[i]. 这样在查询一个区间[l, ...
- BZOJ2120 数颜色 【带修莫队】
BZOJ2120 数颜色 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L R代表询问你从第L支画笔到 ...
- 【BZOJ2453】维护队列/【BZOJ2120】数颜色 分块
[BZOJ2453]维护队列 Description 你小时候玩过弹珠吗? 小朋友A有一些弹珠,A喜欢把它们排成队列,从左到右编号为1到N.为了整个队列鲜艳美观,小朋友想知道某一段连续弹珠中,不同颜色 ...
- 【BZOJ-2453&2120】维护队列&数颜色 分块 + 带修莫队算法
2453: 维护队列 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 653 Solved: 283[Submit][Status][Discuss] ...
- Bzoj 2453: 维护队列 && Bzoj 2120: 数颜色 分块,bitset
2453: 维护队列 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 578 Solved: 247[Submit][Status][Discuss] ...
- BZOJ2453: 维护队列
2453: 维护队列 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 183 Solved: 89[Submit][Status] Descripti ...
随机推荐
- redis主从同步收到以下参数影响
repl-ping-slave-period主从心跳ping的时间间隔.默认10 repl-timeout 从节点超时时间.默认60 repl-backlog-size 主节点保存操作日志的大 ...
- 转:java23种设计模式
以下是学习过程中查询的资料,别人总结的资料,比较容易理解(站在各位巨人的肩膀上,望博主勿究) 概述 设计模式是针对某一类问题的最优解决方案,是从许多优秀的软件系统中总结出的. Java中设计模式(ja ...
- MySQL的Root用户密码
缘由:最近北京市二环内大兴土木,各种挖沟埋线.忽而一纸通令周末断电,故多年不断电的服务器,便令人有了关机后是否还能正常启动的隐忧.其中一台较年迈的服务器中搭载有MySQL数据库.数据库内容本属于外包项 ...
- vue js 保留小数
toFixed(number,fractionDigits ){ //number 保留小数数 //fractionDigits 保留小数位数 var times = Math.pow(10, fra ...
- LVS-DR模式实现调度负载
本篇文章主要梳理一下LVS前端调度过程及用户请求过程 实验架构 准备工作 添加各主机路由联通主机通信 Client IP route add default gw 172.20.17.19 Route ...
- git merge最简洁
一.开发分支(dev)上的代码达到上线的标准后,要合并到 master 分支 git checkout devgit pullgit checkout mastergit merge devgit p ...
- MySQL数据操作(DML)
表结构准备: mysql> CREATE TABLE student( -> sid INT PRIMARY KEY AUTO_INCREMENT, ), -> age INT, ) ...
- SSM框架搭建步骤
首先要导入相关的jar包(spring\spring-core\spring-jdbc\spring-aop\spring-context\spring-webmvc\junit\commons-la ...
- PAT (Basic Level) Practice (中文)1002
1002 写出这个数 (20 分) 读入一个正整数 n,计算其各位数字之和,用汉语拼音写出和的每一位数字. 输入格式: 每个测试输入包含 1 个测试用例,即给出自然数 n 的值.这里保证 n 小于 1 ...
- stm32+lwip(五):以太网帧发送测试
我是卓波,很高兴你来看我的博客. 系列文章: stm32+lwip(一):使用STM32CubeMX生成项目 stm32+lwip(二):UDP测试 stm32+lwip(三):TCP测试 stm32 ...