题目分析:

这题出的好!

首先问题肯定是二分图的最大独立集,如果删去某条匹配边之后独立集是否会变大。

跑出最大流之后流满的边就是匹配边。

如果一个匹配边的两个端点在一个强连通分量里,那这条边删掉之后我们就可以找到一个替代方案使得匹配不变小。

具体的,假设这两个点是x,y。因为两者之间连的是匹配边,那么存在一个路径从t->y->x->s。那只要从s有另一条路径到y或者从x有另一条路径到t那就构成一个强连通分量,我们只考虑s到y的情况。

如果存在一条这样的路,我们会发现每次从X集合跳到Y集合的时候走的是黑边,从Y集合跳到X集合的时候走的是红遍,因为我们的目标节点在Y集合,所以采用匈牙利树的分析方法,黑边总比红边多1,所以可以把这条路径翻转达到我们的目的。

代码:

 #include<bits/stdc++.h>
using namespace std; const int maxm = ,maxn = ; struct edge{int from,to,flow;}edges[maxm];
int num,n,m,arr[maxn];
vector <int> g[maxn];
int color[maxn]; namespace BioGraph{
vector<int> T[maxn];
queue<int> Q;
void pd(){
for(int i=;i<=n;i++){
if(arr[i]) continue;
Q.push(i); arr[i] = ; color[i] = ;
while(!Q.empty()){
int k = Q.front(); Q.pop();
for(int i=;i<T[k].size();i++){
int z = T[k][i];
if(arr[z]) continue;
Q.push(z); color[z] = (color[k]^); arr[z] = ;
}
}
}
}
}
namespace SCC{
int dfn[maxn],low[maxn],scc[maxn],sccnum,cl;
stack<int> sta;
void Tarjan(int now){
low[now] = dfn[now] = ++cl;
sta.push(now);
for(int i=;i<g[now].size();i++){
if(edges[g[now][i]].flow == ) continue;
int to = edges[g[now][i]].to;
if(arr[to]) continue;
if(dfn[to]) low[now] = min(low[now],dfn[to]);
else{Tarjan(to);low[now] = min(low[now],low[to]);}
}
if(low[now]==dfn[now]){
sccnum++;
while(true){
int pi = sta.top(); sta.pop();
arr[pi] = ; scc[pi] = sccnum;
if(now == pi) break;
}
}
}
} void AddEdge(int x,int y,int v){
edges[num++] = (edge){x,y,v};
edges[num++] = (edge){y,x,};
g[x].push_back(num-);
g[y].push_back(num-);
} int dis[maxn],cur[maxn];
queue<int> qq;
int BFS(){
qq.push(); memset(dis,-,sizeof(dis)); dis[] = ;
while(!qq.empty()){
int k = qq.front(); qq.pop();
for(int i=;i<g[k].size();i++){
edge sm = edges[g[k][i]];
if(sm.flow && dis[sm.to]== -){
dis[sm.to] = dis[k]+;
qq.push(sm.to);
}
}
}
return dis[n+];
} int dfs(int x,int a){
if(x == n+ || a == ) return a;
int flow = ,f;
for(int &i=cur[x];i<g[x].size();i++){
edge &xx = edges[g[x][i]];
if(dis[xx.to]>dis[x]&&(f=dfs(xx.to,min(a,xx.flow)))){
xx.flow -= f;
flow += f;
a -= f;
edges[g[x][i]^].flow+=f;
if(a == ) return flow;
}
}
return flow;
} void read(){
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++){
int x,y; scanf("%d%d",&x,&y);
BioGraph::T[x].push_back(y);
BioGraph::T[y].push_back(x);
}
BioGraph::pd();
for(int i=;i<=n;i++){
if(color[i]) {AddEdge(i,n+,);continue;}
else AddEdge(,i,);
for(int j=;j<BioGraph::T[i].size();j++){
int z = BioGraph::T[i][j];
AddEdge(i,z,);
}
}
} vector<pair<int,int> > res;
void work(){
int flow = ;
while(BFS() != -){
memset(cur,,sizeof(cur));
flow += dfs(,);
}
memset(arr,,sizeof(arr));
for(int i=;i<=n+;i++){
if(arr[i]) continue;
SCC::Tarjan(i);
}
for(int i=;i<num;i++){
if(edges[i].from == || edges[i].to == n+) continue;
if(edges[i].from == n+ || edges[i].to == ) continue;
if(color[edges[i].from]) continue;
if(edges[i].flow) continue;
if(SCC::scc[edges[i].from] == SCC::scc[edges[i].to]) continue;
int x=min(edges[i].from,edges[i].to),y=max(edges[i].from,edges[i].to);
res.push_back(make_pair(x,y));
}
sort(res.begin(),res.end());
int z = res.size(); printf("%d\n",z);
for(int i=;i<z;i++){ printf("%d %d\n",res[i].first,res[i].second); }
} int main(){
read();
work();
return ;
}

LOJ2276 [HAOI2017] 新型城市化 【二分图匹配】【tarjan】的更多相关文章

  1. HAOI2017 新型城市化 二分图的最大独立集+最大流+强连通缩点

    题目链接(洛谷):https://www.luogu.org/problemnew/show/P3731 题意概述:给出一张二分图,询问删掉哪些边之后可以使这张二分图的最大独立集变大.N<=10 ...

  2. 【Luogu3731】[HAOI2017]新型城市化(网络流,Tarjan)

    [Luogu3731][HAOI2017]新型城市化(网络流,Tarjan) 题面 洛谷 给定一张反图,保证原图能分成不超过两个团,问有多少种加上一条边的方法,使得最大团的个数至少加上\(1\). 题 ...

  3. 求去掉一条边使最小割变小 HAOI2017 新型城市化

    先求最小割,然后对残量网络跑Tarjan.对于所有满流的边,若其两端点不在同一个SCC中,则这条边是满足条件的. 证明见 来源:HAOI2017 新型城市化

  4. HDU4685:Prince and Princess(二分图匹配+tarjan)

    Prince and Princess Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Othe ...

  5. 洛谷 P3731 [HAOI2017]新型城市化【最大流(二分图匹配)+tarjan】

    我到底怎么建的图为啥要开这么大的数组啊?! 神题神题,本来以为图论出不出什么花来了. 首先要理解'团'的概念,简单来说就是无向图的一个完全子图,相关概念详见度娘. 所以关于团一般都是NP问题,只有二分 ...

  6. Luogu3731 HAOI2017新型城市化(二分图匹配+强连通分量)

    将未建立贸易关系看成连一条边,那么这显然是个二分图.最大城市群即最大独立集,也即n-最大匹配.现在要求的就是删哪些边会使最大匹配减少,也即求哪些边一定在最大匹配中. 首先范围有点大,当然是跑个dini ...

  7. P3731 [HAOI2017]新型城市化(tarjan+网络流)

    洛谷 题意: 给出两个最大团的补图,现在要求增加一条边,使得最大最大团个数增加至少\(1\). 思路: 我们求出团的补图,问题可以转换为:对于一个二分图,选择删掉一条边,能够增大其最大独立集的点集数. ...

  8. Luogu P3731 [HAOI2017]新型城市化

    题目显然可以转化为求每一条边对二分图最大独立集的贡献,二分图最大独立集\(=\)点数\(-\)最大匹配数,我们就有了\(50pts\)做法. 正解的做法是在原图上跑\(Tarjan\),最开始我想复杂 ...

  9. HDU 4685 Prince and Princess 二分图匹配+tarjan

    Prince and Princess 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=4685 Description There are n pri ...

随机推荐

  1. J2SE学习历程

    2014/12/09 1.+两边有字符串的话,则另外的先转换为字符串再连接. int c = 12; System.out.println(“c=” + c); 2.如果a=2,b=a++,先赋值再运 ...

  2. C# 提前异步加载数据

    前言 在我们应用程序中,如果有较大的数据需要从数据库或者本地读取,且是一次性的话,可以提前获取数据并缓存在内存中. 一般处理方法:利用应用程序启动到用户使用功能这一段时间,提前加载数据. 问题来了,因 ...

  3. Java 平台无关性的基石

    Java 在刚刚诞生之初就提出过一个非常著名的口号:"一次编写,到处运行", 这句话充分表达了软件开发人员对冲破平台界限的渴望 在竞争激烈的 IT 领域,各种不同的硬件体系结构和不 ...

  4. 《全栈营销之如何制作个人博客》之一:用什么开发语言和CMS系统

    现在的互联网,已经不再是初级互联时代,是需要全方位营销,自建粉丝池的时代,云码素材从提出全栈营销,到实践分析,总结出全栈营销第一步,先制作一个个人博客是最好的开始.今天开始就跟我一起学习如何制作一个精 ...

  5. dedecms织梦上下页标签和CSS

    模板中的分页部分源码: <div class="pagination-wrapper" style= "font-size:20pt;color:red;text- ...

  6. arcgis api 3.x for js 热力图优化篇-不依赖地图服务(附源码下载)

    前言 关于本篇功能实现用到的 api 涉及类看不懂的,请参照 esri 官网的 arcgis api 3.x for js:esri 官网 api,里面详细的介绍 arcgis api 3.x 各个类 ...

  7. android - TextView单行显示...或者文字左右滚动(走马灯效果)

    条件 TextView单行显示,文字左右滚动(走马灯效果)实现条件: 实现单行设置固定宽度或者设置权重都行 代码 TextView滚动必须写下面几个属性 android:singleLine=&quo ...

  8. JS直接调用C#后台方法(ajax调用)

    1. 先手动引用DLL或者通过NuGet查找引用,这里提供一个AjaxPro.2.dll的下载: 2. 之后的的过程不想写了,网上都大同小异的,直接参考以前大佬写的: AjaxPro2完整入门教程 总 ...

  9. liteos简介(一)

    LiteOS是在2015华为网络大会上华为发布的敏捷网络3.0中的一个轻量级的物联网操作系统,LiteOS体积只有10KB级. 在Hi3559A中,liteos是用于Cortex-A53,用于处理MP ...

  10. Status bar could not find cached time string image. Rendering in-process?

    在开发中,控制台经常输出“Status bar could not find cached time string image. Rendering in-process?” 在 Info.plist ...