hdu5348

题目大意就是有n个点,m条无向边,现让你将这m条无向边改为有向边,使得每个点的|入度-出度|<=1,然后输出m行,1代表第i条边为u-->v,0代表v-->u。逐点逐边搜索即可。如果对于某个点u,它的入度in[u]>=出度out[u],那么就正向搜索,我们找一个与u相连且不等于u的点v,假设点v的in[v]<out[v],那么刚好符合条件,我们就将u,v这条边置为1,使得out[u]++,in[v]++时间复杂度为O(n+m),要达到这个效率,我们用链式向前星来存储边,然后用head[u]=edge[i].next来删除已经访问过的边。但这道题目我不太知道怎么去证明无论如何都不会没有解,即都不会输出-1。

#include<stdio.h>
#include<iostream>
#include<string>
#include<string.h>
#include<algorithm>
#include<iomanip>
#include<vector>
#include<time.h>
#include<queue>
#include<stack>
#include<iterator>
#include<math.h>
#include<stdlib.h>
#include<limits.h>
#include<map>
#include<set>
//#define ONLINE_JUDGE
#define eps 1e-8
#define INF 0x7fffffff
#define FOR(i,a) for((i)=0;i<(a);(i)++)
#define MEM(a) (memset((a),0,sizeof(a)))
#define sfs(a) scanf("%s",a)
#define sf(a) scanf("%d",&a)
#define sfI(a) scanf("%I64d",&a)
#define pf(a) printf("%d\n",a)
#define pfI(a) printf("%I64d\n",a)
#define pfs(a) printf("%s\n",a)
#define sfd(a,b) scanf("%d%d",&a,&b)
#define sft(a,b,num) scanf("%d%d%d",&a,&b,&num)
#define for1(i,a,b) for(int i=(a);i<b;i++)
#define for2(i,a,b) for(int i=(a);i<=b;i++)
#define for3(i,a,b)for(int i=(b);i>=a;i--)
#define MEM1(a) memset(a,0,sizeof(a))
#define MEM2(a) memset(a,-1,sizeof(a))
#define ll __int64
const double PI=acos(-1.0);
template<class T> T gcd(T a,T b){return b?gcd(b,a%b):a;}
template<class T> T lcm(T a,T b){return a/gcd(a,b)*b;}
template<class T> inline T Min(T a,T b){return a<b?a:b;}
template<class T> inline T Max(T a,T b){return a>b?a:b;}
using namespace std;
int n,m;
#define M 110
#define N 10010
#define Mod 1000000007
#define p(x,y) make_pair(x,y)
struct Edge{
	int u,v;
	int next;
}edge[700010];
int head[200010],cnt;
int vis[700010],ans[700010];
int in[200010],out[200010],num[200010];
void Init(){
	cnt=0;
	memset(head,-1,sizeof head);
	memset(vis,0,sizeof vis);
	memset(in,0,sizeof in);
	memset(out,0,sizeof out);
	memset(num,0,sizeof num);
}
void addEdge(int u,int v){
	edge[cnt].u = u;
	edge[cnt].v = v;
	edge[cnt].next = head[u];
	head[u] = cnt++;
}
void dfs1(int u){		//正向搜索
	for(int i=head[u];i!=-1;i=edge[i].next){
		if(vis[i]){
			head[u] = edge[i].next;  //假设i点已经被访问过了,那么修改head[u]=edge[i].next删除u指向i的边
			continue;
		}
		int v = edge[i].v;
		if(u!=v && in[v]>out[v]) continue; //假设非自环且v的入度>出度的话,那就不可以将该边置为1
		vis[i] = vis[i^1] = 1; //将u,v之间的两条边都置为1,说明已经访问过了
		if(i&1) ans[i/2]=0; //如果该边的标号i&1为真,即i为奇数,说明该边原先的输入是(v,u),我们的目的是使得该边变为u-->v
			//那么我们就将其置为0(由题意可得),i/2的结果是那条边真实编号
		else
			ans[i/2]=1;
		out[u]++; //边的方向为(u,v),故u的出度+1
		in[v]++;	//v的入度+1
		head[u] = edge[i].next; //删除该边
		dfs1(v);	//继续从v往下搜索
		break;	//每次只遍历一种情况,因为我们要判断in[i]和out[i]的大小才能决定搜索的方向
	}
}
void dfs2(int u){		//反向搜索,原理跟上面一样
	for(int i=head[u];i!=-1;i=edge[i].next){
		if(vis[i]){
			head[u] = edge[i].next;
			continue;
		}
		int v = edge[i].v;
		if(u!=v && out[v]>in[v]) continue;
		if(i&1) ans[i/2]=1;
		else
			ans[i/2]=0;
		vis[i] = vis[i^1] = 1;
		in[u]++;
		out[v]++;
		head[u] = edge[i].next;
		dfs2(v);
		break;
	}
}
int main() {
#ifndef ONLINE_JUDGE
	freopen("in.txt", "r", stdin);
//	freopen("out.txt", "w", stdout);
#endif
   int t;
   sf(t);
   while(t--){
	   sfd(n,m);
	   Init();
	   int u,v;
	   for(int i=0;i<m;i++){
		   sfd(u,v);
		   addEdge(u,v);
		   addEdge(v,u);
		   num[u]++;
		   num[v]++;		//统计某个点的度数
	   }
	   for(int i=1;i<=n;i++){
		   while(in[i]+out[i]<num[i]){ //目前点的入度+出度<该点的总度数
			   if(in[i]>=out[i])	//该点的入度>出度,那么正向搜索
				   dfs1(i);
			   else
				   dfs2(i);
		   }
	   }
	   for(int i=0;i<m;i++)
		   printf("%d\n",ans[i]);
   }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值