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决 ...
随机推荐
- 【8】学习C++之this指针
在学习类的时候,我们可以考虑到一种情况: class Array { public: Array(int len); ~Array(); void setLen(int len) { len=len; ...
- java List深拷贝示例
示例一 import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.ArrayList; import java ...
- Django(二)模板
一.模板概念 1.Django通过模板动态生成html 2.模板的加载位置 模板一般建立在templates文件夹中,全局路径的设置在settings.py中 DIRS:决定了整个项目的模板路径的 ...
- python基础2--if,while,for,逻辑运算
1.1 条件控制语句 1.if - elif - else 2.常用操作运算符 < > >= <= == != 3.if elif 后面一定要有条件 else后面没有条件 1 ...
- 13、vue如何解决跨域问题
开发环境:配置config文件夹中index.js文件: proxyTable: { '/api': { target: 'http://10.10.1.242:8245',//后端地址 // sec ...
- Numpy和Pandas的使用入门
Numpy Numpy基本数据结构 np.array()函数接受一个多维list,返回对应纬度的矩阵 vector = np.array([1, 2, 3, 4]) matrix = np.array ...
- ubuntu 开启ip包转发
1. 开启IP转发 //临时 # echo "1"> /proc/sys/net/ipv4/ip_forward //永久 # nano /etc/sysctl.conf n ...
- Golang Web应用 创建docker镜像笔记(win 平台)
记录的是 本地编译好了再创建容器镜像的方法 ,这样子生成的镜像文件比较小,方便分发部署 win 平台需要设置golang交叉编译 生成linux可执行文件 CMD下: Set GOOS="l ...
- Linux基础:sort命令总结
本文只总结一些常用的用法,更详细的说明见man sort和sort --help. sort命令 sort命令用于串联排序指定文件并将结果写到标准输出. sort可以指定按照何种排序规则进行排序,如按 ...
- nginx.conf配置项
环境:centos7 nginx1.16.1 以下配置均在配置文件中进行:/etc/nginx/nginx.conf 1.设置工作进程的所有者和所属组 user 所有者 所属组: 设置后要在操作 ...