假期的宿舍 二分图

Code:

#include<cstdio>
#include<cstring>
#include<vector>
#include<queue>
#include<algorithm>
using namespace std;
const int maxn=300;
const int INF=1000000+233;
int C[maxn][maxn],is[maxn],home[maxn],idx[maxn];
int s,t;
struct Edge{
	int from,to,cap;
	Edge(int u,int v,int c):from(u),to(v),cap(c){}
};
struct Dicnic{
	vector<Edge>edges;
	vector<int>G[maxn];
	queue<int>Q;
	int d[maxn],vis[maxn],current[maxn];
	void init()
	{
		edges.clear();
		for(int i=0;i<maxn;++i)G[i].clear();
		memset(d,0,sizeof(d));
	    memset(vis,0,sizeof(vis));
	    memset(current,0,sizeof(current));
	}
	void add_edge(int u,int v,int c)
	{
		edges.push_back(Edge(u,v,c));
		edges.push_back(Edge(v,u,0));
		int m=edges.size();
		G[u].push_back(m-2);
		G[v].push_back(m-1);
	}
	int BFS()
	{
		memset(vis,0,sizeof(vis));
		Q.push(s);vis[s]=1,d[s]=0;
		while(!Q.empty())
		{
			int u=Q.front();Q.pop();
			int siz=G[u].size();
			for(int i=0;i<siz;++i)
			{
				Edge r=edges[G[u][i]];
				if(!vis[r.to]&&r.cap>0)
				{
					vis[r.to]=1,d[r.to]=d[u]+1;
					Q.push(r.to);
				}
			}
		}
		if(vis[t])return 1;
		return 0;
	}
	int dfs(int x,int cur)
	{
		if(x==t)return cur;
		int f,flow=0;
		int siz=G[x].size();
		for(int i=current[x];i<siz;++i)
		{
			current[x]=i;
			int u=G[x][i];
			Edge r=edges[u];
			if(d[r.to]==d[x]+1&&r.cap>0)
			{
				f=dfs(r.to,min(cur,r.cap));
				if(f>0)
				{
					cur-=f,flow+=f;
					edges[u].cap-=f,edges[u^1].cap+=f;
				}
			}
			if(cur==0)break;
		}
		return flow;
	}
	int maxflow()
	{
		int flow=0;
		while(BFS())
		{
			memset(current,0,sizeof(current));
			flow+=dfs(s,INF);
		}
		return flow;
	}
}op;
void init()
{
	memset(is,0,sizeof(is));
	memset(home,0,sizeof(home));
	memset(C,0,sizeof(C));
}
int main()
{
	//freopen("in.txt","r",stdin);
	int T,n;
	scanf("%d",&T);
	while(T--)
	{
		op.init();
		init();
		int num=0,cc=0;
		scanf("%d",&n);
		for(int i=1;i<=n;++i)
		{
			int a;scanf("%d",&a);
			if(a)
			{
				is[i]=1,num+=1,idx[i]=num;
			}
		}
		for(int i=1;i<=n;++i)
		{
			int a;scanf("%d",&a);
			if(is[i]&&a)home[i]=1,cc+=1;
		}
		for(int i=1;i<=n;++i)
			for(int j=1;j<=n;++j)scanf("%d",&C[i][j]);
		s=0,t=188;
		for(int i=1;i<=n;++i)
		{
			if(is[i])op.add_edge(n+idx[i],t,1);
			if(!home[i])
			{
				op.add_edge(s,i,1);
				if(is[i])op.add_edge(i,idx[i]+n,1);
				for(int j=1;j<=n;++j)
				{
					if(C[i][j]&&is[j])op.add_edge(i,n+idx[j],1);
				}
			}
		}
		int ans=op.maxflow();
		if(ans==n-cc)printf("^_^\n");
		else printf("T_T\n");
	}
	return 0;
}

  

原文地址:https://www.cnblogs.com/guangheli/p/9934554.html

时间: 2024-11-11 12:32:51

假期的宿舍 二分图的相关文章

BZOJ 1433 假期的宿舍 二分图匹配

这道题目不难,二分图匹配建模比较明显.加油吧!相信自己.(自己写的,好开心,40毫秒,比ccz略快). 尽管算法模版是抄一本书上的,但这次很明显我是背出来的.不算抄. 1 #include<cstdio> 2 #include<iostream> 3 #include<vector> 4 #include<queue> 5 #include<cstring> 6 #define rep(i,j,k) for(int i = j; i <=

[BZOJ1433][ZJOI2009]假期的宿舍 二分图匹配

题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1433 首先留在学校的学生向自己的床连边. 要住在学校里的人向认识的学生的床连边. 跑二分图匹配,看匹配的数量是否等于住在学校的人数. 1 #include<cstdio> 2 #include<cstring> 3 #include<algorithm> 4 using namespace std; 5 int inline readint(){ 6 int N

BZOJ1433 [ZJOI2009]假期的宿舍 二分图匹配 匈牙利算法

原文链接http://www.cnblogs.com/zhouzhendong/p/8372785.html 题目传送门 - BZOJ1433 题解 我们理一理题目. 在校的学生,有自己的床,还可以睡朋友的床. 离校的学生,不占床. 外来的学生,只能睡朋友的床. 然后就是一个裸的二分图匹配了. 代码 #include <cstring> #include <cstdlib> #include <cmath> #include <cstdio> #includ

P2055 [ZJOI2009]假期的宿舍 - 二分图最大匹配

把人和床分开考虑,题目说每个人只能睡和自己直接认识的人的床,就是一种边的关系,但是并不是人与人,实际上人与人之间连边是很难处理的,但是如果把人和床连边,就是一张二分图,左右两边分别是不同的东西,然后求一下最大匹配就好了 没思路的时候换换角度,看能不能搞出什么"新东西"来 注意多组数据不超时的情况下能用memset尽量用,有时候感觉某个数组可以不清空但实际上是需要清空的 还有注意连边的时候没那么简单...看注释吧 #include <algorithm> #include &

假期的宿舍——二分图模板一个

因为题目并没有给出自己认识自己的数据,所以需要手动从每一个学生向他的床上连一条边,找这个错误找了一个小时. 1 #include<algorithm> 2 #include<iostream> 3 #include<cstring> 4 #include<cstdio> 5 #include<vector> 6 using namespace std; 7 const int N=77; 8 int n,p,beds,res,bed[N],mx[

bzoj1433:[ZJOI2009]假期的宿舍

明显的二分图最大匹配. #include<cstdio> #include<cstring> #include<cctype> #include<algorithm> #include<bitset> using namespace std; #define rep(i,s,t) for(int i=s;i<=t;i++) #define dwn(i,s,t) for(int i=s;i>=t;i--) #define clr(x,c

bzoj1433: [ZJOI2009]假期的宿舍

1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2286  Solved: 969[Submit][Status][Discuss] Description Input Output Sample Input 131 1 00 1 00 1 11 0 01 0 0 Sample Output ˆ ˆ HINT 对于30% 的数据满足1 ≤ n ≤ 12.对于100% 的数据满足1 ≤ n ≤ 50,1 ≤

bzoj1433[ZJOI2009]假期的宿舍(匈牙利)

1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2544  Solved: 1074 [Submit][Status][Discuss] Description Input Output Sample Input 1 3 1 1 0 0 1 0 0 1 1 1 0 0 1 0 0 Sample Output ˆ ˆ HINT 对于30% 的数据满足1 ≤ n ≤ 12.对于100% 的数据满足1 ≤ n

假期的宿舍

假期的宿舍 题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1433 二分图匹配(最大流) 做到Candy(http://acm.split.hdu.edu.cn/showproblem.php?pid=4322)这道题,用A*写了半天TLE了,去网上搜题解,都是网络流的做法.问艾神用搜索做该怎么剪枝,艾神说他写的书上有讲很多剪枝策略,然而最快三天后书才能到....,所以就先试试网络流了. 之前没怎么接触过网络流,看了半天才看懂一些,于是