题目描述

二阶堂真红给了你一个长为n的序列a,有m次操作

1.把区间[l,r]中大于x的数减去x

2.查询区间[l,r]中x的出现次数

题解

做YNOI真**爽。。。

我们发现这道题的操作非常诡异,把大于x的数减去x。

想一想这个操作会带来什么,会使这个数列的极差减小x(如果有大于x的数的话)。

然后如果我们能以至多O(x)的代价是极差缩小x的话,那么它的总复杂度是不会超过O(n)的。

然后我们发现log数据结构没法维护这种东西。

于是就用到了分块,每个块中维护每种数字出现的最早位置,然后把相同的数用并查集并起来。

对于修改操作。

我们讨论一下,x-0和maxnum-x哪个大,x-0大的话,就把小于x的数加上x,并且打个标记。

否则就把大于x的减掉x,这个用并查集搞就好了。

边角直接暴力就好了,注意每次暴力之前要把所有数都搞成当前数字的值。

注意数组越界。。。

代码

#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize ("Ofast")
#pragma GCC optimize ("Ofast","inline","unroll-loops")
#pragma GCC optimize ("no-stack-protector")
#include<cstdio>
#include<vector>
#include<cmath>
#include<cctype>
#define R register
#define N 100009
#define M 318
using namespace std;
typedef long long ll;
int n,m,n1,f[N],head[M][N],a[N],be[N],ma[M],la[M],cnt[N];
inline int rd(){
int x=;char c=getchar();
while(!isdigit(c)){c=getchar();}
while(isdigit(c)){x=(x<<)+(x<<)+(c^);c=getchar();}
return x;
}
int find(int x){return f[x]=f[x]==x?x:find(f[x]);}
inline int mn(const int &x,const int &y){
return x>y?y:x;
}
inline int mx(const int &x,const int &y){
return x>y?x:y;
}
inline void build(const int &x){
ma[x]=;
int maxn=mn(n,x*n1);
for(R int i=(x-)*n1+;i<=maxn;++i)cnt[i]=,f[i]=i;
for(R int i=(x-)*n1+;i<=maxn;++i){
ma[x]=mx(ma[x],a[i]);
if(!head[x][a[i]])head[x][a[i]]=i,cnt[i]=;else{f[i]=head[x][a[i]];cnt[head[x][a[i]]]++;}
}
}
inline void pushdown(const int &x){
int maxn=mn(n,x*n1);
for(R int i=(x-)*n1+;i<=maxn;++i)a[i]=a[find(i)],head[x][a[i]]=;
for(R int i=(x-)*n1+;i<=maxn;++i)a[i]-=la[x];la[x]=;
}
inline void upd(const int &x,const int &pre,const int &now){
ma[x]=mx(ma[x],now);
if(!head[x][now]){
head[x][now]=head[x][pre];head[x][pre]=;a[head[x][now]]=now;
return;
}
f[head[x][pre]]=head[x][now];cnt[head[x][now]]+=cnt[head[x][pre]];cnt[head[x][pre]]=;head[x][pre]=;
}
int main(){
n=rd();m=rd();int opt,l,r,x;n1=sqrt(n);
for(R int i=;i<=n;++i)be[i]=(i-)/n1+,f[i]=i,a[i]=rd();
for(R int i=;i<=be[n];++i)build(i);
while(m--){
opt=rd();l=rd();r=rd();x=rd();
if(opt==){
if(be[l]==be[r]){
pushdown(be[l]);
for(R int i=l;i<=r;++i)if(a[i]>x)a[i]-=x;
build(be[l]);
}
else{
pushdown(be[l]);pushdown(be[r]);
for(R int i=l;i<=be[l]*n1;++i)if(a[i]>x)a[i]-=x;
for(R int i=(be[r]-)*n1+;i<=r;++i)if(a[i]>x)a[i]-=x;
build(be[l]);build(be[r]);
for(int i=be[l]+;i<be[r];++i){
if(ma[i]-la[i]>*x){
for(R int j=la[i]+;j<=la[i]+x;++j)if(head[i][j])upd(i,j,j+x);
la[i]+=x;
}
else{
for(R int j=la[i]++x;j<=ma[i];++j)if(head[i][j])upd(i,j,j-x);
ma[i]=mn(ma[i],la[i]+x);
}
}
}
}
else{
int ans=;
if(be[l]==be[r]){
for(R int i=l;i<=r;++i)if(a[find(i)]-la[be[l]]==x)ans++;
}
else{
for(R int i=l;i<=be[l]*n1;++i)if(a[find(i)]-la[be[l]]==x)ans++;
for(R int i=(be[r]-)*n1+;i<=r;++i)if(a[find(i)]-la[be[r]]==x)ans++;
for(R int i=be[l]+;i<be[r];++i)if(x+la[i]<N)ans+=cnt[head[i][x+la[i]]];
}
printf("%d\n",ans);
}
}
return ;
}

[Ynoi2018]五彩斑斓的世界的更多相关文章

  1. 洛谷P4117 [Ynoi2018]五彩斑斓的世界 [分块,并查集]

    洛谷 Codeforces 又是一道卡常题-- 思路 YNOI当然要分块啦. 分块之后怎么办? 零散块暴力,整块怎么办? 显然不能暴力改/查询所有的.考虑把相同值的用并查集连在一起,这样修改时就只需要 ...

  2. [YNOI2018]五彩斑斓的世界&CF896E(分块+并查集)

    由于晚上比赛二连(Atcoder&codeforces),外加复习学考,所以暂时没时间写了. 贴个O(n√n)的分块代码,洛谷和cf上都过了,但垃圾bzoj卡不过去,不改了. #include ...

  3. 解题:由乃OI 2018 五彩斑斓的世界

    题面 写在前面的扯淡: 分块的总体学习告一段落,这算是分块集中学习的最后一题么:以后当然也可能会写,就是零零散散的题了=.= 在洛谷上搜ynoi发现好像只有这道题和 由乃OI 2018 未来日记 是分 ...

  4. [bzoj 5143][Ynoi 2018]五彩斑斓的世界

    传送门 Descroption 给了你一个长为n的序列a,有m次操作 1.把区间[l,r]中大于x的数减去x 2.查询区间[l,r]中x的出现次数 Solution 分块 对于每个块,我们都分别维护: ...

  5. Solution -「Ynoi 2018」「洛谷 P4117」五彩斑斓的世界

    \(\mathcal{Description}\)   Link.   给定序列 \(\{a_n\}\),处理 \(m\) 次操作: 给定 \(l,r,x\),把 \([l,r]\) 内所有 \(&g ...

  6. NOIP前的刷题记录

    因为这几天要加油,懒得每篇都来写题解了,就这里记录一下加上一句话题解好了 P4071 [SDOI2016]排列计数   组合数+错排 loj 6217 扑克牌 暴力背包 P2511 [HAOI2008 ...

  7. tricks - 思维

    编辑 目录 tricks 系列 随机的性质 bitmask 建图 最基本的 黑白染色 Kruskal重构树 并查集维护值域 带根号的数三元环 根号分治 调和级数哈希 多属性哈希 时光倒流 时光反复横跳 ...

  8. [Ynoi2018]未来日记

    "望月悲叹的最初分块" (妈呀这名字好中二啊(谁叫我要用日本轻小说中的东西命名真是作死)) 这里就直接挂csy的题解了,和我的不太一样,但是大概思路还是差不多的,我的做法是和“五彩 ...

  9. ynoi2018

    题解: 全分块是啥操作啊.. 而且都好难.. 1.未来日记 这个比较简单 对每个块开个线段树维护权值 $n\sqrt{n}logn$ 这个会炸空间 并不能做... 但还是说一下做法 首先考虑分块 然后 ...

随机推荐

  1. C语言检测指定文件是否存在的代码

    内容之余,将做工程过程中比较常用的一些内容片段珍藏起来,下面资料是关于C语言检测指定文件是否存在的内容,希望能对小伙伴们有所用. #include <stdbool.h> #include ...

  2. [Android][Recovery] Recovery下找不到sdcard路径

    做升级的时候,把更新包拷贝到sd卡中,然后调用接口进行重启升级 wossoneri.github.io File update_file = new File("/sdcard/update ...

  3. java基础知识总结一:

      四种内部类 直接抛出异常 单例模式: 懒汉式单例.饿汉式单例.登记式单例     []关于内部类:  []关于异常: 直接捕捉并抛出异常:不需要给异常添加名字: if(i>10)throw ...

  4. (办公)plug-in org.eclipse.jdt.ui was unable to load class org.eclipse.jdt.internal

    今天上午开发环境遇到这个问题,解决方案如下,(解决了之后,项目并没有丢失.) 因为Eclipse的这个plug-in org.eclipse.jdt.ui was unable to load cla ...

  5. Ext.grid.panel 改变某一行的字体颜色

    grid.getStore().addListener('load', handleGridLoadEvent); function handleGridLoadEvent(store, record ...

  6. python 通过元类控制类的创建

    一.python中如何创建类? 1. 直接定义类 class A: a = 'a' 2. 通过type对象创建 在python中一切都是对象 在上面这张图中,A是我们平常在python中写的类,它可以 ...

  7. Xamarin移动开发的优点和缺点

    在考虑iOS或Android应用程序开发时,我们大多数人会首先考虑Objective-C vs Swift和Java.作为本地技术堆栈,当涉及到iOS和Android应用程序开发时,它们自然是最常用的 ...

  8. Nagle 算法

    1. Nagel算法        TCP/IP协议中,无论发送多少数据,总是要在数据前面加上协议头,同时,对方接收到数据,也需要发送ACK表示确认.为了尽可能的利用网络带宽,TCP总是希望尽可能的发 ...

  9. RabbitMQ权限控制原理

    我们在使用MQ搭建系统的时候,经常要开放队列给外接系统访问.外接系统的稳定性是不可控的.为了防止外接系统不稳定导致误操作破坏了MQ的配置或数据,需要对MQ做比较精细的权限控制. 我的需求是这样的: 我 ...

  10. Centos6系列Bond配置方法

    在Windows Server平台因业务需求经常会用到NIC双网卡绑定,同样Linux平台下用于网络负载均衡及网络冗余会用到bond模式. Bond模式:0-6,即7种模式. 模式一:mod=0 ,即 ...