【刷题】BZOJ 3546 [ONTAK2010]Life of the Party
Description
一个舞会有N个男孩(编号为1..N)和M个女孩(编号为1..M),一对男女能够组成一对跳舞当且仅当他们两个人互相认识。
我们把一种人定义成这个舞会的life:当且仅当如果他(她)不参加这个舞会,那么能够同时配对的最大舞伴对数会下降。
现在知道男生和女生之间的认识关系,需要你求出男生和女生中的是这个舞会的life的人的编号。
Input
第一行3个整数N,M,K,表示N个男生,M个女生,K对关系。
接下来K行,每行两个整数a_i b_i,表示第a_i个男生和第b_i个女生相互认识。
Output
首先输出所有男生中是这个舞会的life的男生的编号,一行一个,从小到大输出,然后输出女生的。
Sample Input
4 4 4
2 1
3 2
4 3
4 4
Sample Output
2
3
4
1
2
HINT
N,M<=104,K<=105
Solution
一个匹配点 \(s\) 能变成非匹配点,当且仅当从这个点出发能找一条以匹配边出发的交替链,使得终点是某个未盖点 \(t\) 。由于链长为偶数, \(t\) 和 \(s\) 属于同一侧
所以对于左边,匹配边从右到左,非匹配边从左到右,从左侧某个未盖点出发DFS,给那些到达的点打标记,最终左侧没有标记的点就是关键点
对于右边则相反
#include<bits/stdc++.h>
#define ui unsigned int
#define ll long long
#define db double
#define ld long double
#define ull unsigned long long
#define boy(x) x
#define girl(x) n+x
const int MAXN=10000+10,MAXK=100000+10,inf=0x3f3f3f3f;
int e=1,n,m,k,s,t,to[MAXK<<2],nex[MAXK<<2],cap[MAXK<<2],beg[MAXN<<1],match[MAXN],vis[MAXN<<1],clk,qe,qto[MAXK<<2],qnex[MAXK<<2],qbeg[MAXN<<1],level[MAXN<<1],out[MAXK<<2],cur[MAXN<<1];
std::queue<int> q;
template<typename T> inline void read(T &x)
{
T data=0,w=1;
char ch=0;
while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
if(ch=='-')w=-1,ch=getchar();
while(ch>='0'&&ch<='9')data=((T)data<<3)+((T)data<<1)+(ch^'0'),ch=getchar();
x=data*w;
}
template<typename T> inline void write(T x,char ch='\0')
{
if(x<0)putchar('-'),x=-x;
if(x>9)write(x/10);
putchar(x%10+'0');
if(ch!='\0')putchar(ch);
}
template<typename T> inline void chkmin(T &x,T y){x=(y<x?y:x);}
template<typename T> inline void chkmax(T &x,T y){x=(y>x?y:x);}
template<typename T> inline T min(T x,T y){return x<y?x:y;}
template<typename T> inline T max(T x,T y){return x>y?x:y;}
inline void insert(int x,int y,int z)
{
to[++e]=y;
out[e]=x;
nex[e]=beg[x];
beg[x]=e;
cap[e]=1;
to[++e]=x;
out[e]=y;
nex[e]=beg[y];
beg[y]=e;
cap[e]=0;
}
inline bool bfs()
{
memset(level,0,sizeof(level));
level[s]=1;
q.push(s);
while(!q.empty())
{
int x=q.front();
q.pop();
for(register int i=beg[x];i;i=nex[i])
if(!level[to[i]]&&cap[i])
{
level[to[i]]=level[x]+1;
q.push(to[i]);
}
}
return level[t];
}
inline int dfs(int x,int maxflow)
{
if(x==t||!maxflow)return maxflow;
vis[x]=clk;
int res=0,f;
for(register int &i=cur[x];i;i=nex[i])
if((vis[x]^vis[to[i]])&&cap[i]&&level[to[i]]==level[x]+1)
{
f=dfs(to[i],min(maxflow,cap[i]));
res+=f;
cap[i]-=f;
cap[i^1]+=f;
maxflow-=f;
if(!maxflow)break;
}
vis[x]=0;
return res;
}
inline int Dinic()
{
int res=0;
while(bfs())++clk,memcpy(cur,beg,sizeof(cur)),res+=dfs(s,inf);
return res;
}
inline void qinsert(int x,int y)
{
qto[++qe]=y;
qnex[qe]=qbeg[x];
qbeg[x]=qe;
}
inline void exdfs(int x,int sx)
{
vis[x]=1;
if(sx?(x>n):(x<=n))match[sx?x-n:x]=0;
for(register int i=qbeg[x];i;i=qnex[i])
if(!vis[qto[i]])exdfs(qto[i],sx);
}
int main()
{
read(n);read(m);read(k);
for(register int i=1;i<=k;++i)
{
int u,v;
read(u);read(v);
insert(boy(u),girl(v),1);
}
s=n+m+1,t=s+1;
for(register int i=1;i<=n;++i)insert(s,boy(i),1);
for(register int i=1;i<=m;++i)insert(girl(i),t,1);
Dinic();
memset(vis,0,sizeof(vis));
for(register int i=2;i<=(k<<1);i+=2)
if(!cap[i])qinsert(to[i],out[i]),match[out[i]]=1;
else qinsert(out[i],to[i]);
for(register int i=1;i<=n;++i)
if(!match[i]&&!vis[i])exdfs(i,0);
for(register int i=1;i<=n;++i)
if(match[i])write(i,'\n');
qe=0;
memset(vis,0,sizeof(vis));
memset(qbeg,0,sizeof(qbeg));
memset(match,0,sizeof(match));
for(register int i=2;i<=(k<<1);i+=2)
if(!cap[i])qinsert(out[i],to[i]),match[to[i]-n]=1;
else qinsert(to[i],out[i]);
for(register int i=1;i<=m;++i)
if(!match[i]&&!vis[i+n])exdfs(i+n,1);
for(register int i=1;i<=m;++i)
if(match[i])write(i,'\n');
return 0;
}
【刷题】BZOJ 3546 [ONTAK2010]Life of the Party的更多相关文章
- BZOJ.3546.[ONTAK2010]Life of the Party(二分图匹配 ISAP)
题目链接 题意:求哪些点一定在最大匹配中. 这儿写过,再写一遍吧. 求哪些点不一定在最大匹配中.首先求一遍最大匹配,未匹配点当然不一定在最大匹配中. 设一个未匹配点为A,如果存在边A-B,且存在匹配边 ...
- BZOJ 3546 [ONTAK2010]Life of the Party (二分图最大匹配必须点)
题解:给出一个二分图,问你取点哪个点会使得二分图匹配数减少. 解法:其实就是问二分图匹配的必须点.先对初始二分图做一次最大匹配. 现在考虑左边点,看残余网络上的边重新构图:如果是匹配边那么就从右往左连 ...
- 【刷题】BZOJ 2407 探险
Description 探险家小T好高兴!X国要举办一次溶洞探险比赛,获奖者将得到丰厚奖品哦!小T虽然对奖品不感兴趣,但是这个大振名声的机会当然不能错过! 比赛即将开始,工作人员说明了这次比赛的规则: ...
- 【刷题】BZOJ 4543 [POI2014]Hotel加强版
Description 同OJ3522 数据范围:n<=100000 Solution dp的设计见[刷题]BZOJ 3522 [Poi2014]Hotel 然后发现dp的第二维与深度有关,于是 ...
- 【刷题】BZOJ 4316 小C的独立集
Description 图论小王子小C经常虐菜,特别是在图论方面,经常把小D虐得很惨很惨. 这不,小C让小D去求一个无向图的最大独立集,通俗地讲就是:在无向图中选出若干个点,这些点互相没有边连接,并使 ...
- 【刷题】BZOJ 4176 Lucas的数论
Description 去年的Lucas非常喜欢数论题,但是一年以后的Lucas却不那么喜欢了. 在整理以前的试题时,发现了这样一道题目"求Sigma(f(i)),其中1<=i< ...
- BZOJ第一页刷题计划
BZOJ第一页刷题计划 已完成:67 / 90 [BZOJ1000]A+B Problem:A+B: [BZOJ1001][BeiJing2006]狼抓兔子:最小割: [BZOJ1002][FJOI2 ...
- 【刷题】BZOJ 2260 商店购物
Description Grant是一个个体户老板,他经营的小店因为其丰富的优惠方案深受附近居民的青睐,生意红火.小店的优惠方案十分简单有趣.Grant规定:在一次消费过程中,如果您在本店购买了精制油 ...
- 【刷题】BZOJ 4566 [Haoi2016]找相同字符
Description 给定两个字符串,求出在两个字符串中各取出一个子串使得这两个子串相同的方案数.两个方案不同当且仅当这两个子串中有一个位置不同. Input 两行,两个字符串s1,s2,长度分别为 ...
随机推荐
- AOSP 设置编译输出目录
export OUT_DIR=/media/caoxinyu/TomasYu/out 注意:export OUT_DIR= OUT_DIR 后面直接跟= ,不要有空格.否则报错.
- spring源码-bean之增强初始化-3
一.ApplicationContext的中文意思是“应用上下文”,它继承自BeanFactory接口,除了包含BeanFactory的所有功能之外,在国际化支持.资源访问(如URL和文件).事件传播 ...
- Mac安装php和redis扩展
Mac上有特定的包管理工具homebrew,也叫brew,这里的php安装用的就是brew 1安装php brew install php@7.0. brw安装会自动管理依赖,所以不用你一个个先安装依 ...
- spark 执行架构
术语定义 Application:Spark Application的概念和Hadoop MapReduce中的类似,指的是用户编写的Spark应用程序,包含了一个Driver 功能的代码和分布在集群 ...
- yield学习
如果要控制内存占用,最好不要用list来保存中间结果,而是通过iterable对象(range, xrange, generator等)来迭代. yield 使函数变为generator,返回对象 ...
- mybati缓存机制之二级缓存配置
二级缓存配置 在MyBatis的配置文件中开启二级缓存. <setting name="cacheEnabled" value="true"/> 在 ...
- 适配chrome65最新selenium-chromedriver
网盘地址:https://pan.baidu.com/s/1BmdwRgD96IL32-3FTFxPSg 密码: 2vg6
- TW实习日记:第28天
同前两天一样,等接口,开发,调试接口.重复地做着低级代码得搬运工作,确实挺没意思的.怪不得有些人一直说写低级代码很无聊,没有创造性和成就感.31号准备溜了,还是好好复习准备秋招吧. 挖坑清单: Vue ...
- 【springmvc+mybatis项目实战】杰信商贸-3.需求分析与数据库建模
开发步骤需求:生产厂家信息维护基础表FACTORY_C 1.业务需求:a)<需求说明书> 1)描述业务功能 生产厂家模块 功能:为在购销合同模块中的货物信息和附件信 ...
- @meida 媒体查询
示例 @meida 媒体查询 在进行书写的时候需要考虑到加载顺序和样式权重使用meida响应式实现不同宽度布局示例 常用工具 https://mydevice.io 参考链接 https://deve ...