BZOJ1116: [POI2008]CLO

时间:2023-03-09 19:53:02
BZOJ1116: [POI2008]CLO

1116: [POI2008]CLO

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 565  Solved: 303
[Submit][Status]

Description

Byteotia城市有n个 towns m条双向roads. 每条 road 连接 两个不同的 towns ,没有重复的road. 你要把其中一些road变成单向边使得:每个town都有且只有一个入度

Input

第一行输入n m.1 <= n<= 100000,1 <= m <= 200000 下面M行用于描述M条边.

Output

TAK或者NIE 常做POI的同学,应该知道这两个单词的了...

Sample Input

4 5
1 2
2 3
1 3
3 4
1 4
BZOJ1116: [POI2008]CLO

Sample Output

TAK
BZOJ1116: [POI2008]CLO
上图给出了一种连接方式.

题解:

又被并查集虐哭了。。。

hzwer:

首先我们可以推出一个性质,当且仅当某一个连通块中没有环存在输出NIE
于是我们可以用并查集把这题水过。。
给每个连通块的根一个标记
合并俩个连通块时只要任意一个连通块带有标记新的连通块就带有标记
如果一条边的俩个点已经在同一个连通块内了,直接将根打上标记即可
然后最后扫一遍,如果哪个连通块没有标记的话输出NIE
否则最后输出TAK
-----------------------------------------------------------------------------------------------------------
代码:
 #include<cstdio>
#include<cstdlib>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<vector>
#include<map>
#include<set>
#include<queue>
#include<string>
#define inf 1000000000
#define maxn 100000+100
#define maxm 500+100
#define eps 1e-10
#define ll long long
#define pa pair<int,int>
#define for0(i,n) for(int i=0;i<=(n);i++)
#define for1(i,n) for(int i=1;i<=(n);i++)
#define for2(i,x,y) for(int i=(x);i<=(y);i++)
using namespace std;
inline int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=*x+ch-'';ch=getchar();}
return x*f;
}
int n,m,fa[maxn];
bool mark[maxn];
inline int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);};
int main()
{
freopen("input.txt","r",stdin);
freopen("output.txt","w",stdout);
n=read();m=read();
for1(i,n)fa[i]=i;
for1(i,m)
{
int x=find(read()),y=find(read());
if(x==y)mark[x]=;else fa[x]=y,mark[y]=mark[x]||mark[y];
}
for1(i,n)if(!mark[find(i)]){cout<<"NIE"<<endl;return ;}
cout<<"TAK"<<endl;
return ;
}

网络流看起来是可行的,但T成翔啊。。。

 #include<cstdio>
#include<cstdlib>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<vector>
#include<map>
#include<set>
#include<queue>
#include<string>
#define inf 1000000000
#define maxn 1000000
#define maxm 1000000
#define eps 1e-10
#define ll long long
#define pa pair<int,int>
#define for0(i,n) for(int i=0;i<=(n);i++)
#define for1(i,n) for(int i=1;i<=(n);i++)
#define for2(i,x,y) for(int i=(x);i<=(y);i++)
using namespace std;
inline int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=*x+ch-'';ch=getchar();}
return x*f;
}
int n,m,s,t,maxflow,tot=,head[maxn],q[maxm],d[maxn],h[maxn],cur[maxn];
bool v[maxn];
struct edge{int from,next,go,v;}e[*maxm];
void ins(int x,int y,int z){e[++tot].go=y;e[tot].v=z;e[tot].next=head[x];head[x]=tot;}
void insert(int x,int y,int z){ins(x,y,z);ins(y,x,);}
bool bfs()
{
for(int i=s;i<=t;i++)h[i]=-;
int l=,r=;q[]=s;h[s]=;
while(l<r)
{
int x=q[++l];
for(int i=head[x];i;i=e[i].next)
if(e[i].v&&h[e[i].go]==-)
{
h[e[i].go]=h[x]+;q[++r]=e[i].go;
}
}
return h[t]!=-;
}
int dfs(int x,int f)
{
if(x==t) return f;
int tmp,used=,i;
for(int i=head[x];i;i=e[i].next)
if(e[i].v&&h[e[i].go]==h[x]+)
{
tmp=dfs(e[i].go,min(e[i].v,f-used));
e[i].v-=tmp;if(e[i].v)cur[x]=i;
e[i^].v+=tmp;used+=tmp;
if(used==f)return f;
}
if(!used) h[x]=-;
return used;
}
void dinic()
{
maxflow=;
while(bfs())
{
for (int i=s;i<=t;i++)cur[i]=head[i];maxflow+=dfs(s,inf);
}
}
int main()
{
freopen("input.txt","r",stdin);
freopen("output.txt","w",stdout);
n=read();m=read();s=;t=m+n+n+;
for1(i,m)
{
int x=read(),y=read();
insert(s,i,);
insert(i,m+x,);insert(i,m+y,);
insert(m+x,m+n+y,);insert(m+y,m+n+x,);
}
for1(i,n)insert(m+n+i,t,);
dinic();
if(maxflow==n)cout<<"TAK"<<endl;else cout<<"NIE"<<endl;
return ;
}