5098: [BZOJ1098][POI2007]办公楼biu
[BZOJ1098][POI2007]办公楼biu
一个看上去非常玄学的代码
const int N=1e5+10,M=2e6+10;
int n,m;
int fa[N];
int Find(int x){ return fa[x]==x?x:fa[x]=Find(fa[x]); }
vector <int> G[N];
typedef vector <int> ::iterator iter;
void merge(int x,int y){
x=Find(x),y=Find(y);
if(x==y) return;
vector <int> tmp;tmp.clear();
if(G[x].size()>G[y].size()) swap(x,y);
int p=0;
for(reg int i=0;i<(int)G[x].size();i++){
while(p<(int)G[y].size()&&G[y][p]<G[x][i]) p++;
if(p>=(int)G[y].size()||G[y][p]!=G[x][i]) {
tmp.push_back(G[x][i]);
G[x].erase(G[x].begin()+i);
i--;
}
}
fa[y]=x;
rep(i,0,tmp.size()-1) merge(x,tmp[i]);
}
int cnt[N];
int ans[N],ac;
bool go[N];
void Sort(vector <int> &V){
if(V.size()<20000) return sort(V.begin(),V.end());
memset(go,0,sizeof go);
rep(i,0,V.size()) go[V[i]]=true;
V.clear();
rep(i,1,n) if(go[i]) V.push_back(i);
}
int num[N];
bool cmp(int x,int y){
return G[x].size()<G[y].size();
}
int main(){
n=rd(),m=rd();
rep(i,1,n) fa[i]=num[i]=i;
rep(i,1,m) {
int u=rd(),v=rd();
G[u].push_back(v);
G[v].push_back(u);
}
rep(i,1,n) Sort(G[i]);
sort(num+1,num+n+1,cmp);
rep(i,1,n) if(Find(num[i])==num[i]) {
memset(go,0,sizeof go);
rep(j,0,G[num[i]].size()-1) go[G[num[i]][j]]=1;
rep(j,1,n) if(num[i]!=j&&!go[j]) merge(num[i],j);
}
rep(i,1,n) cnt[Find(i)]++;
rep(i,1,n) if(cnt[i]) ans[++ac]=cnt[i];
sort(ans+1,ans+ac+1);
printf("%d\n",ac);
rep(i,1,ac) printf("%d",ans[i]),putchar(i==ac?'\n':' ');
}
我这个代码大概是在讲这样的:
按照出度排序,没有连边的点必须合并,最后找集合个数
但是合并的时候要注意用y的答案更新x,这样y就不用再跑一次了(这里的总复杂度较为诡异)
更新过程:遍历x的边,如果y的边中没有x的这条边,那么就将其删去,并且将这个点合并
首先根据鸽巢原理,出度最小的点最大出度必然是\(\frac{m}{n}\)
它的边数也是\(\frac{m}{n}\)
每次和其他的点合并,这个复杂度是 \(\frac{m}{n} \cdot m\) (事实上是远远跑不到的!因为每次都会删去大量的边!)
复杂度似乎是有保证的
然后由于合并完之后也就剩下不到\(\frac{m}{n}\)个点了,所以这里就算你暴力\(O(n)\)合并也没关系
最后是你们担心的\(\frac{m}{n} \cdot m\)会TLE
瞪眼一看当\(m=2e6,n\)较小时就T了
哪你n较小时直接跑\(O(n^2)\)不就是了 吐舌头
注意时限是20s的这个代码还根本跑不满,所以自然不用担心!
tips: vector可以用链表优化更棒
5098: [BZOJ1098][POI2007]办公楼biu的更多相关文章
- BZOJ1098: [POI2007]办公楼biu
从问题可以看出是求补图的连通块及点数 但补图太大.所以考虑缩小规模. 当一个点归属于一个连通块后,它以后就不需要了.所以可以用链表,删去这个点,也就减小了规模. 一个点开始bfs,每个点只会进队一次, ...
- 【链表】Bzoj1098[POI2007]办公楼biu
Description FGD开办了一家电话公司.他雇用了N个职员,给了每个职员一部手机.每个职员的手机里都存储有一些同事的电话号码.由于FGD的公司规模不断扩大,旧的办公楼已经显得十分狭窄,FGD决 ...
- BZOJ1098 POI2007 办公楼biu 【链表+bfs】
Description FGD开办了一家电话公司.他雇用了N个职员,给了每个职员一部手机.每个职员的手机里都存储有一些同事的电话号码.由于FGD的公司规模不断扩大,旧的办公楼已经显得十分狭窄,FGD决 ...
- bzoj 1098 [POI2007]办公楼biu bfs+补图+双向链表
[POI2007]办公楼biu Time Limit: 20 Sec Memory Limit: 162 MBSubmit: 1543 Solved: 743[Submit][Status][Di ...
- 【BZOJ1098】[POI2007]办公楼biu
题目一开始看以为和强联通分量有关,后来发现是无向边,其实就是求原图的补图的联通块个数和大小.学习了黄学长的代码,利用链表来优化,其实就是枚举每一个人,然后把和他不相连的人都删去放进同一个联通块里,利用 ...
- 【BZOJ】1098: [POI2007]办公楼biu(补图+bfs+链表)
http://www.lydsy.com/JudgeOnline/problem.php?id=1098 显然答案是补图连通块..... 想到用并查集...可是连补图的边都已经...n^2了...怎么 ...
- 【刷题】BZOJ 1098 [POI2007]办公楼biu
Description FGD开办了一家电话公司.他雇用了N个职员,给了每个职员一部手机.每个职员的手机里都存储有一些同事的 电话号码.由于FGD的公司规模不断扩大,旧的办公楼已经显得十分狭窄,FGD ...
- bzoj 1098 [POI2007] 办公楼 biu
# 解题思路 画画图可以发现,只要是两个点之间没有相互连边,那么就必须将这两个人安排到同一个办公楼内,如图所示: 那,我们可以建立补图,就是先建一张完全图,然后把题目中给出的边都删掉,这就是一张补图, ...
- [POI2007]办公楼biu
Description FGD开办了一家电话公司.他雇用了N个职员,给了每个职员一部手机.每个职员的手机里都存储有一些同事的电话号码.由于FGD的公司规模不断扩大,旧的办公楼已经显得十分狭窄,FGD决 ...
随机推荐
- windows7下安装msys2
系统: windows 7 首先需要msys2的安装包,可以去官网下载安装包官网地址: http://www.msys2.org/本次下载的是 msys2-x86_64-20190524.exe 注意 ...
- Java 数组实例——将阿拉伯数字转换为最大写
题目:将阿拉伯数字转换为最大写,比如1234转换为壹仟贰佰叁拾肆. package my_package; public class Transform { private String[] arr1 ...
- sizeof()计算
本节包含sizeof()计算结构体,位域,数组,字符串,指针,c++中的class等类型的大小,sizeof()计算的大小都是以字节为单位. 一 计算基本类型的长度 sizeof(char): 1 s ...
- socket系统化入门
1.简单socket完成消息发送与接收 服务端: package com.wfd360.com.socket; import java.io.*; import java.net.ServerSock ...
- 我的windows开发环境设定与日常使用指南
目录 开发相关的软件包安装.设定 Visual Studio 默认设定 鼠标右键添加"在此处打开cmd"选项 git gvim notepad++ VSCode-Insider C ...
- nohup-长期运行进程
我们经常会碰到这样的问题,用 telnet/ssh 登录了远程的 Linux 服务器,运行了一些耗时较长的任务, 结果却由于网络的不稳定导致任务中途失败. 解决办法: 当用户注销(logout)或者网 ...
- Android Studio Gradle 配置问题
中国境内可以直接访问 dl.google.com 不会被墙. 由于种种原因导致的Gradle出现各类型配置问题在各大论坛.提问区已经是新人常问的问题了,自己也遇到很恶心的几个问题: Valid cer ...
- python的包管理软件Conda的用法
创建自己的虚拟环境 conda create -n learn python= 切换环境: activate learn 查看所有环境: conda env list 安装第三方包: conda in ...
- Java8的Stream API使用
前言 这次想介绍一下Java Stream的API使用,最近在做一个新的项目,然后终于可以从老项目的祖传代码坑里跳出来了.项目用公司自己的框架搭建完成后,我就想着把JDK版本也升级一下吧(之前的项目, ...
- cmds系统数据库源端大表数据更新优化
cmds系统数据库源端大表数据更新优化 以下脚本可以用于将表按照rowid范围分区,获得指定数目的rowid Extent区间(Group sets of rows in the table into ...