洛谷P4117 [Ynoi2018]五彩斑斓的世界 [分块,并查集]
又是一道卡常题……
思路
YNOI当然要分块啦。
分块之后怎么办?
零散块暴力,整块怎么办?
显然不能暴力改/查询所有的。考虑把相同值的用并查集连在一起,这样修改时就只需要枚举值了。
然而每次修改的\(x\)特别小时仍然复杂度爆炸,发现大于\(x\)的减去\(x\)等价于小于等于\(x\)的加上\(x\),然后整体减去\(x\)。
那么,设一个块的最大值为\(mx\),则
- \(2x\geq mx\)时枚举\(x<v\leq mx\),把\(v\)的并查集连到\(v-x\)上,复杂度\(O(mx-x)\),\(mx\)变为\(x\)。(即\(mx-=mx-x\))
- 否则,枚举\(v\leq x\),把\(v\)的并查集连到\(v+x\)上,然后整体减去\(x\),复杂度\(O(x)\),\(mx-=x\)。
你是否发现一件事:在每一块上花\(O(x)\)的时间,就可以使\(mx\)减小\(x\)。也就是说,每一块花费的最多时间是\(O(mx)\),总共就是\(O(\sqrt m \times mx)\)。
也就是说,这种仿佛暴力的算法,复杂度是正确的。
然而,毒瘤出题人既卡空间,又卡时间,所以要AC还是有些麻烦的。
不过CF上给了\(512MB,3000ms\),可以去那里。
CF的题似乎被暴力+pragma艹过去了。出题人活该
代码
没怎么卡常的代码,CF上能过:
#include<bits/stdc++.h>
namespace my_std{
using namespace std;
#define pii pair<int,int>
#define fir first
#define sec second
#define MP make_pair
#define rep(i,x,y) for (int i=(x);i<=(y);i++)
#define drep(i,x,y) for (int i=(x);i>=(y);i--)
#define go(x) for (int i=head[x];i;i=edge[i].nxt)
#define sz 100010
#define S 320
typedef long long ll;
typedef double db;
mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
template<typename T>inline T rnd(T l,T r) {return uniform_int_distribution<T>(l,r)(rng);}
template<typename T>inline void read(T& t)
{
t=0;char f=0,ch=getchar();double d=0.1;
while(ch>'9'||ch<'0') f|=(ch=='-'),ch=getchar();
while(ch<='9'&&ch>='0') t=t*10+ch-48,ch=getchar();
if(ch=='.'){ch=getchar();while(ch<='9'&&ch>='0') t+=d*(ch^48),d*=0.1,ch=getchar();}
t=(f?-t:t);
}
template<typename T,typename... Args>inline void read(T& t,Args&... args){read(t); read(args...);}
void file()
{
#ifndef ONLINE_JUDGE
freopen("a.txt","r",stdin);
#endif
}
#ifdef mod
ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x%mod) if (y&1) ret=ret*x%mod;return ret;}
ll inv(ll x){return ksm(x,mod-2);}
#else
ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x) if (y&1) ret=ret*x;return ret;}
#endif
// inline ll mul(ll a,ll b){ll d=(ll)(a*(double)b/mod+0.5);ll ret=a*b-d*mod;if (ret<0) ret+=mod;return ret;}
}
using namespace my_std;
int n,m;
int a[sz];
int fa[sz],size[sz];
int getfa(int x){return x==fa[x]?x:fa[x]=getfa(fa[x]);}
int blo,pos[sz];
struct hh
{
int L,R;
int tag,mx;
int f[sz];
int val(int x){return a[getfa(x)]-tag;} // the real value of position x
void rebuild()
{
rep(i,L,R) fa[i]=size[i]=0;
tag=mx=0;
rep(i,L,R)
if (!f[a[i]]) f[a[i]]=fa[i]=i,size[i]=1;
else fa[i]=f[a[i]],++size[fa[i]];
rep(i,L,R) mx=max(mx,a[i]);
}
void modify(int x)
{
if (x>=mx) return;
if (2*x>=mx)
{
rep(i,x+1+tag,mx+tag) if (f[i])
{
int y=i-x;
if (f[y]) fa[f[i]]=f[y],size[f[y]]+=size[f[i]];
else f[y]=f[i],a[f[y]]-=x;
f[i]=0;
}
mx=x;
}
else
{
drep(i,x+tag,1+tag) if (f[i])
{
int y=i+x;
if (f[y]) fa[f[i]]=f[y],size[f[y]]+=size[f[i]];
else f[y]=f[i],a[f[y]]+=x;
f[i]=0;
}
tag+=x;mx-=x;
}
}
int query(int x){return x+tag<sz&&f[x+tag]?size[f[x+tag]]:0;}
}b[S];
void init()
{
blo=sqrt(n);
rep(i,1,n)
{
pos[i]=(i-1)/blo+1;
b[pos[i]].R=i;
if (pos[i]!=pos[i-1]) b[pos[i]].L=i;
}
rep(i,1,pos[n]) b[i].rebuild();
}
int tmp[sz];
void change_(int l,int r,int x)
{
int p=pos[l];
rep(i,b[p].L,b[p].R) b[p].f[a[getfa(i)]]=0;
rep(i,b[p].L,b[p].R) tmp[i]=b[p].val(i);
rep(i,b[p].L,b[p].R) a[i]=tmp[i];
rep(i,l,r) if (a[i]>x) a[i]-=x;
b[p].rebuild();
}
void change(int l,int r,int x)
{
int pl=pos[l],pr=pos[r];
if (pl==pr) return change_(l,r,x);
change_(l,b[pl].R,x),change_(b[pr].L,r,x);
if (pl==pr+1) return;
++pl,--pr;
rep(i,pl,pr) b[i].modify(x);
}
int query_(int l,int r,int x)
{
int ret=0,p=pos[l];
rep(i,l,r)
ret+=(b[p].val(i)==x);
return ret;
}
int query(int l,int r,int x)
{
int pl=pos[l],pr=pos[r];
if (pl==pr) return query_(l,r,x);
int ret=query_(l,b[pl].R,x)+query_(b[pr].L,r,x);
if (pl+1==pr) return ret;
++pl,--pr;
rep(i,pl,pr) ret+=b[i].query(x);
return ret;
}
int main()
{
file();
read(n,m);
rep(i,1,n) read(a[i]);
init();
while (m--)
{
int z,x,l,r;read(z,l,r,x);
if (z==1) change(l,r,x);
else printf("%d\n",query(l,r,x));
}
return 0;
}
卡常之后的代码,洛谷上可以过:
注意:不保证此代码任何时刻都可通过。我是半夜提交的。
#pragma GCC optimize(1)
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize("Ofast,no-stack-protector,unroll-loops,fast-math")
#pragma GCC target("sse,sse2,sse3,ssse3,sse4.1,sse4.2,avx,avx2,popcnt,tune=native")
#include<bits/stdc++.h>
namespace my_std{
using namespace std;
#define rep(i,x,y) for (register int i=x;i<=y;++i)
#define drep(i,x,y) for (register int i=x;i>=y;--i)
#define sz 100010
#define S 320
typedef long long ll;
typedef double db;
char buf[1<<20],*p1=buf,*p2=buf;
inline char getchar(){return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<20,stdin),p1==p2)?EOF:*p1++;}
inline void read(int& t)
{
t=0;char ch=getchar();
while(ch>'9'||ch<'0') ch=getchar();
while(ch<='9'&&ch>='0') t=t*10+ch-48,ch=getchar();
}
char sr[800000],z[10];int C=-1,Z=0;
inline void Ot(){fwrite(sr,1,C+1,stdout),C=-1;}
void print(register int x)
{
if(x<0)sr[++C]='-',x=-x;
while(z[++Z]=x%10+48,x/=10);
while(sr[++C]=z[Z],--Z);sr[++C]='\n';
}
// void file()
// {
// #ifndef ONLINE_JUDGE
// freopen("a.txt","r",stdin);
// #endif
// }
}
using namespace my_std;
int n,m;
int a[sz];
int fa[sz],size[sz];
int getfa(int x){return x==fa[x]?x:fa[x]=getfa(fa[x]);}
int blo,pos[sz];
struct hh
{
int L,R;
int tag,mx;
int f[sz];
void rebuild()
{
rep(i,L,R) fa[i]=size[i]=0;
tag=mx=0;
rep(i,L,R)
if (!f[a[i]]) f[a[i]]=fa[i]=i,size[i]=1;
else fa[i]=f[a[i]],++size[fa[i]];
rep(i,L,R) mx=max(mx,a[i]);
}
void modify(int x)
{
if (x>=mx) return;
if (2*x>=mx)
{
rep(i,x+1+tag,mx+tag) if (f[i])
{
int y=i-x;
if (f[y]) fa[f[i]]=f[y],size[f[y]]+=size[f[i]];
else f[y]=f[i],a[f[y]]-=x;
f[i]=0;
}
mx=x;
}
else
{
drep(i,x+tag,1+tag) if (f[i])
{
int y=i+x;
if (f[y]) fa[f[i]]=f[y],size[f[y]]+=size[f[i]];
else f[y]=f[i],a[f[y]]+=x;
f[i]=0;
}
tag+=x;mx-=x;
}
}
inline int query(register int x){return x+tag<sz&&f[x+tag]?size[f[x+tag]]:0;}
}b[S];
void init()
{
blo=sqrt(n);
rep(i,1,n) pos[i]=(i-1)/blo+1;
rep(i,1,pos[n]) b[i].L=i*blo-blo+1,b[i].R=min(n,i*blo),b[i].rebuild();
}
int tmp[sz];
void change_(int l,int r,int x)
{
int p=pos[l],tg=b[p].tag;
rep(i,b[p].L,b[p].R) b[p].f[a[getfa(i)]]=0;
rep(i,b[p].L,b[p].R) tmp[i]=a[getfa(i)]-tg;
rep(i,b[p].L,b[p].R) a[i]=tmp[i];
rep(i,l,r) if (a[i]>x) a[i]-=x;
b[p].rebuild();
}
void change(int l,int r,int x)
{
int pl=pos[l],pr=pos[r];
if (pl==pr) return change_(l,r,x);
change_(l,b[pl].R,x),change_(b[pr].L,r,x);
if (pl==pr+1) return;
++pl,--pr;
rep(i,pl,pr) b[i].modify(x);
}
int query_(int l,int r,int x)
{
int ret=0,p=pos[l],X=b[p].tag+x;
rep(i,l,r)
ret+=(a[getfa(i)]==X);
return ret;
}
int query(int l,int r,int x)
{
int pl=pos[l],pr=pos[r];
if (pl==pr) return query_(l,r,x);
int ret=query_(l,b[pl].R,x)+query_(b[pr].L,r,x);
if (pl+1==pr) return ret;
++pl,--pr;
rep(i,pl,pr) ret+=b[i].query(x);
return ret;
}
int main()
{
// file();
// clock_t t=clock();
read(n);read(m);
rep(i,1,n) read(a[i]);
init();
register int z,x,l,r;
rep(_,1,m)
{
read(z);read(l);read(r);read(x);
if (z==1) change(l,r,x);
else print(query(l,r,x));
}
Ot();
// cout<<(clock()-t)/1000.0;
// return 0;
}
洛谷P4117 [Ynoi2018]五彩斑斓的世界 [分块,并查集]的更多相关文章
- Solution -「Ynoi 2018」「洛谷 P4117」五彩斑斓的世界
\(\mathcal{Description}\) Link. 给定序列 \(\{a_n\}\),处理 \(m\) 次操作: 给定 \(l,r,x\),把 \([l,r]\) 内所有 \(&g ...
- 洛谷P1991无线通讯网[kruskal | 二分答案 并查集]
题目描述 国防部计划用无线网络连接若干个边防哨所.2 种不同的通讯技术用来搭建无线网络: 每个边防哨所都要配备无线电收发器:有一些哨所还可以增配卫星电话. 任意两个配备了一条卫星电话线路的哨所(两边都 ...
- 洛谷 P2024 食物链 POJ 1182 Label:并查集Turbo
题目描述 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A 吃 B,B 吃 C,C 吃 A. 现有 N 个动物,以 1 - N 编号.每个动物都是 A,B,C 中的一种,但是我 ...
- 洛谷P3402 【模板】可持久化并查集(可持久化线段树,线段树)
orz TPLY 巨佬,题解讲的挺好的. 这里重点梳理一下思路,做一个小小的补充吧. 写可持久化线段树,叶子节点维护每个位置的fa,利用每次只更新一个节点的特性,每次插入\(logN\)个节点,这一部 ...
- [洛谷P2024/POJ1182]食物链 - 带偏移量的并查集(2)
Description 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A 吃 B,B吃 C,C 吃 A. 现有 N 个动物,以 1 - N 编号.每个动物都是 A,B,C 中的 ...
- 货车运输-洛谷-1967-LCA+最大生成树(kruskal(并查集))
传送门 一道:LCA+最大生成树 个人认为把这两个的板子写好(并熟练掌握了之后)就没什么难的 (但我还是de了好久bug)qwq 最大生成树:其实就是最小生成树的变形 我用的是kruskal (个人觉 ...
- 洛谷P4768 [NOI2018]归程(可持久化并查集,最短路)
闲话 一个蒟蒻,在网络同步赛上进行了这样的表演-- T2组合计数不会,T3字符串数据结构不会,于是爆肝T1 一开始以为整个地图都有车,然后写了2h+的树套树,终于发现样例过不去 然后写可持久化并查集D ...
- 洛谷 P1525 【关押罪犯】种类并查集
题解 P1525 [关押罪犯]:种类并查集 前言: 在数据结构并查集中,种类并查集属于扩展域并查集一类. 比较典型的题目就是:食物链(比本题难一些,有三个种类存在) 首先讲一下本题的贪心,这个是必须要 ...
- 洛谷P3402 【模板】可持久化并查集 [主席树,并查集]
题目传送门 可持久化并查集 n个集合 m个操作 操作: 1 a b 合并a,b所在集合 2 k 回到第k次操作之后的状态(查询算作操作) 3 a b 询问a,b是否属于同一集合,是则输出1否则输出0 ...
随机推荐
- select实现简单TCP通信(ubuntu 18.04)
一.服务器程序(server.c) #include <stdio.h> #include <unistd.h> #include <stdlib.h> #incl ...
- [SYSS-2018-033]: Fujitsu Wireless Keyboard Set LX901 - Keystroke Injection Vulnerability
风险简介: [SYSS-2018-033]:富士通无线键盘组LX901 -击键注入漏洞 风险报告ID: sys - 2018 - 033 产品:无线键盘套件LX901 制造商:富士通 受影响版本:型号 ...
- [C++]2-2 韩信点兵
/* 韩信点兵 相传韩信才智过人,从不直接清点自己军队的人数,只要让士兵先后以三人一排.五人一排.七人一排地变换队 形,而他每次只掠一眼队伍的排尾就知道总人数了.输入多组数据,每组数据包含3个非负整数 ...
- bebugger调试理解commonJS原理
上面图片是bebugger一个导入的模块,使用vscode可以轻松的看到调用栈,通过断点调试进入断点 Mode函数的原型里面有一个require方法,函数里面有Module._load()加载模块,传 ...
- C语言库函数syslog
参考链接: http://blog.csdn.net/jiangxinyu/article/details/1473356
- shiro 和 跨域过滤器冲突
在web.xml 中 cros要写在shiro 的配置前面,如下,不然会先过shiro 过滤器就产生跨域问题, 参考web.xml 的过滤器加载顺序 <!-- 跨域请求 --> <f ...
- Maven多模块项目管理小结
原文地址:http://blog.csdn.net/whuslei/article/details/7989102 题记 最近刚完成一个用Maven构建的Web项目,看了一些Maven方面的书,比如& ...
- MySql cmd下的学习笔记 —— 有关建立表的操作(有关于数据类型)
(01)建表的过程实际上是 声明字段 的过程 一. 列类型(字段): 存储同样的数据时,不同的列类型,所占据的空间和效率是不一样的,这就是建表时要考虑的意义. 二.MySQL三大列类型 数值型 ...
- 1173 - The Vindictive Coach
题目大意:n个不同身高的队员和教练的按照身高排成波浪形……每个人按照身高由低到高编号,其中第m个是教练,他必须在第一个,如果条件允许,排第二的要比m低,如果条件不允许,即其余人都比教练高,则要让差距尽 ...
- nova 命令管理虚拟机
nova命令管理虚拟机: $ nova list #查看虚拟机$ nova stop [vm-name]或[vm-id] #关闭虚拟机$ nova start [vm-name]或[vm-id] #启 ...