AT4505-[AGC029F]Construction of a tree【构造题,hall定理,网络流】
正题
题目链接:https://www.luogu.com.cn/problem/AT4505
题目大意
给出\(n\)个点和\(n-1\)个点集\(U_i\),每个点集中选择两个点连边使得该图是一棵树。求方案。
\(n\in[1,10^5],\sum_{i=1}^{n-1} |U_i|\in[1,2*10^5]\)
解题思路
冬令营上讲的题目,现在来写。(而且好像我记得课上讲的做法是\(bitset\)的,还是时间久了我记岔了?)
第一眼看上去直觉像是\(hall\)定理但还是不会。
hall定理:\(2*n\)个点的二分图匹配,如果满足任意\(k\)个点都连接了不少于\(k\)个点的话,那么这张图就有完全匹配。
先套一下试试,发现满足条件的图对于它的每个子图\(S\)满足该子图是一个森林。
换句话说对于任意一个\(U\)的集合\(T\),\(G(T)\)表示选出的边连接的节点个数,那么一定有\(G(T)\geq |T|+1\)
回顾一下\(hall\)定理发现是不是很像。
可以先给每个点集选出一个各不同的点(也就是跑一次匹配),如果选不出来那么显然无解。
然后考虑另一个点的选择,从没有被选择的那个点入手,这个点可以选择任何一个包含它的点集连接出去,然后就从下一个点集开始,直到回溯回来选择下一个。如果最后能够遍历所有点就是合法的。
考虑一下正确性,如果它不能遍历所有点那么没有被遍历的点集\(T\)无论怎么连接外面,就一定有一个环不满足\(G(T)\geq |T|+1\)。如果它能遍历所有点,那么我们已经构造出一个方案,显然合法。
时间复杂度\(O(\sum_{i=1}^{n-1}|E|\sqrt n+n)\)
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
const int N=4e5+10,inf=1e9;
struct node{
int to,next,w;
}a[N*2];
int n,s,t,tot=1,cnt,ans;
int ls[N],p[N],b[N],dep[N],cur[N];
queue<int> q;bool v[N];
void addl(int x,int y,int w){
a[++tot].to=y;a[tot].next=ls[x];ls[x]=tot;a[tot].w=w;
a[++tot].to=x;a[tot].next=ls[y];ls[y]=tot;a[tot].w=0;
return;
}
bool bfs(){
for(int i=1;i<=t;i++)
cur[i]=ls[i],dep[i]=0;
while(!q.empty())q.pop();q.push(s);
dep[s]=1;
while(!q.empty()){
int x=q.front();q.pop();
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(dep[y]||!a[i].w)continue;
dep[y]=dep[x]+1;
if(y==t)return 1;
q.push(y);
}
}
return 0;
}
int dinic(int x,int flow){
if(x==t)return flow;
int rest=0,k;
for(int &i=cur[x];i;i=a[i].next){
int y=a[i].to;
if(dep[x]+1!=dep[y]||!a[i].w)continue;
rest+=(k=dinic(y,min(a[i].w,flow-rest)));
a[i].w-=k;a[i^1].w+=k;
if(rest==flow)return rest;
}
if(!rest)dep[x]=0;
return rest;
}
void dfs(int x){
cnt++;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(v[y])continue;
b[y]=x;v[y]=1;dfs(p[y]);
}
}
int main()
{
scanf("%d",&n);s=2*n;t=s+1;
for(int i=1;i<n;i++){
int m;scanf("%d",&m);
for(int j=1;j<=m;j++){
int x;scanf("%d",&x);
addl(x,i+n,1);
}
}
for(int i=1;i<=n;i++)addl(s,i,1);
for(int i=1;i<n;i++)addl(i+n,t,1);
while(bfs())
ans+=dinic(s,inf);
if(ans<n-1)return puts("-1")&0;
for(int x=n+1;x<s;x++)
for(int i=ls[x];i;i=a[i].next)
if(a[i].w){p[x]=a[i].to;break;}
v[s]=1;
for(int i=ls[s];i;i=a[i].next)
if(a[i].w)dfs(a[i].to);
if(cnt<n)return puts("-1")&0;
for(int x=n+1;x<s;x++)
printf("%d %d\n",p[x],b[x]);
return 0;
}
AT4505-[AGC029F]Construction of a tree【构造题,hall定理,网络流】的更多相关文章
- @atcoder - AGC029F@ Construction of a tree
目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定 N - 1 个 {1, 2, ..., N} 的子集,第 ...
- 【构造题 贪心】cf1041E. Tree Reconstruction
比赛时候还是太慢了……要是能做快点就能上分了 Monocarp has drawn a tree (an undirected connected acyclic graph) and then ha ...
- HDU 5573 Binary Tree 构造
Binary Tree 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5573 Description The Old Frog King lives ...
- AIM Tech Round 4 (Div. 1) C - Upgrading Tree 构造 + 树的重心
C - Upgrading Tree 我发现我构造题好弱啊啊啊. 很明显能想到先找到重心, 然后我们的目标就是把所有点接到重心的儿子上,让重心的儿子子树变成菊花图, 这个先把重心到儿子的边连到 i , ...
- cf251.2.C (构造题的技巧)
C. Devu and Partitioning of the Array time limit per test 1 second memory limit per test 256 megabyt ...
- hdu4671 Backup Plan ——构造题
link:http://acm.hdu.edu.cn/showproblem.php?pid=4671 其实是不难的那种构造题,先排第一列,第二列从后往前选. #include <iostrea ...
- Educational Codeforces Round 7 D. Optimal Number Permutation 构造题
D. Optimal Number Permutation 题目连接: http://www.codeforces.com/contest/622/problem/D Description You ...
- Codeforces 482 - Diverse Permutation 构造题
这是一道蛮基础的构造题. - k +(k - 1) -(k - 2) 1 + k , 1 , k , 2, ....... ...
- BZOJ 3097: Hash Killer I【构造题,思维题】
3097: Hash Killer I Time Limit: 5 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 963 Solved: 36 ...
随机推荐
- Spring-Boot注入自定义properties文件配置
创建wzq.properties wzq.properties注入User实体类中 @PropertySource(value = "classpath:wzq.properties&quo ...
- Spring boot集成Redis实现sessions共享时,sessions过期时间问题分析
Springboot鼓励零配置的方式,帮你做好大部分重复劳动的事,好到不能再好:具体的Redis安装方法和Springboot集成Redis方法,可以去搜索相关文章或参考该文章http://www.c ...
- 服务器程序动态加载自定义jar包的过程
需求: 用过hive的都知道,可以自定义hive的一个udf jar,然后将这个jar add到hive服务端,就会加载这个jar实现用户自定义逻辑.现在的需求就是实现这么一个服务端所做的事情! 场景 ...
- 【C语言】第1章 程序设计与C语言
第1章 程序设计与C语言 程序:一组计算机能识别和执行的 指令. 计算机语言:人和计算机交流信息的.计算机和人都能识别的语言 C语言允许用两种注释方式: //:单行注释 可单独占一行 可出现在一行中其 ...
- Kickstart部署多系统
原文转自:https://www.cnblogs.com/itzgr/p/10029587.html作者:木二 目录 一 准备 1.1 完整架构:Kickstart+DHCP+HTTP+TFTP+PX ...
- [bug]spring项目通过反射测试私有方法时,注入对象异常
背景 遇到问题:在进行Spring单元测试编写时,发现被测方法是一个私有方法,无法直接通过注入对象调用 解决思路:首先想到通过反射获取该私有方法的访问权限,并传入注入对象,最终调用对象的私有方法. 出 ...
- 【Python机器学习实战】决策树与集成学习(五)——集成学习(3)GBDT应用实例
前面对GBDT的算法原理进行了描述,通过前文了解到GBDT是以回归树为基分类器的集成学习模型,既可以做分类,也可以做回归,由于GBDT设计很多CART决策树相关内容,就暂不对其算法流程进行实现,本节就 ...
- RabbitMQ之消息模式1
消息100%的投递 消息如何保障100%的投递成功? 什么是生产端的可靠性投递? 保障消息的成功发出 保障MQ节点的成功接收 发送端收到MQ节点(Broker)确认应答 完善的消息进行补偿机制 BAT ...
- 这款打怪升级的小游戏,7 年前出生于 GitHub 社区,如今在谷歌商店有 8 万人打了满分
今天我在 GitHub 摸鱼寻找新的"目标"时,发现了一个开源项目是 RougeLike 类的角色扮演游戏「破碎版像素地牢」(Shattered Pixel Dungeon)类似魔 ...
- Appium自动化(16) - 使用手机浏览器进行自动化测试
如果你还想从头学起Appium,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1693896.html 前言 前面我都讲的都是针对 app ...