【链接】 我是链接,点我呀:)

【题意】

在这里输入题意

【题解】

把每个人都分为左边和右边两个人
xi,yi
如果第i个人不回家或者是外校学生
那么它可以和他认识的人连一条容量为1的边(前提是这个认识的人是本校的学生)
(从左边连向右边
然后源点向每个不回家的本校人或者外校人连一条容量为1的边。
(边的终点是左边的人
每个不是外校的人向汇点T连一条容量为1的边。
(边的起点是右边的人
做一下最大流就可以了
看看最大流和需要安排床位的人的人数相同不相同就可以了
(其实也就是二分图匹配,每条匹配就对应了给每个人分配一个床位的过程
(左边是需要分配床位的那些人,右边是有床的人。。

【代码】

#include <bits/stdc++.h>
#define LL long long
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define all(x) x.begin(),x.end()
#define pb push_back
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
using namespace std; const double pi = acos(-1);
const int dx[4] = {0,0,1,-1};
const int dy[4] = {1,-1,0,0};
const int N = 50; int n,r[N+10],l[N+10],cnt,a[N+10][N+10]; struct abc{
int en,nex;
LL flow;
}; int s,t,deep[N*2+20];
int fir[N*2+10],tfir[N*2+10],totm;
abc bian[2*N*N+4*N+10];
queue <int> dl; void add(int x,int y,LL cost){
bian[totm].nex = fir[x];
fir[x] = totm;
bian[totm].en = y,bian[totm].flow = cost;
totm++; bian[totm].nex = fir[y];
fir[y] = totm;
bian[totm].en = x,bian[totm].flow = 0;
totm++;
} bool bfs(int s,int t){
dl.push(s);
memset(deep,255,sizeof deep);
deep[s] = 0; while (!dl.empty()){
int x = dl.front();
dl.pop();
for (int temp = fir[x]; temp!= -1 ;temp = bian[temp].nex){
int y = bian[temp].en;
if (deep[y]==-1 && bian[temp].flow>0){
deep[y] = deep[x] + 1;
dl.push(y);
}
}
}
return deep[t]!=-1;
} LL dfs(int x,int t,LL limit){
if (x == t) return limit;
if (limit == 0) return 0;
LL cur,f = 0;
for (int temp = tfir[x];temp!=-1;temp = bian[temp].nex){
tfir[x] = temp;
int y = bian[temp].en;
if (deep[y] == deep[x] + 1 && (cur = dfs(y,t,min(limit,(LL)bian[temp].flow))) ){
f += cur;
limit -= cur;
bian[temp].flow -= cur;
bian[temp^1].flow += cur;
if (!limit) break;
}
}
return f;
} int main(){
#ifdef LOCAL_DEFINE
freopen("rush_in.txt", "r", stdin);
#endif
ios::sync_with_stdio(0),cin.tie(0);
int T;
cin>>T;
while (T--){
int students = 0;
//如果不是在校学生 那么右边没有点
//因为他没有床位,不用考虑这个点的入度
totm = 0;
memset(fir,255,sizeof fir);
memset(r,0,sizeof r);
memset(l,0,sizeof l);
cnt = 0;
cin >> n;
for (int i = 1;i <= n;i++){
int x;
cin >> x;
if (x==1) r[i]=++cnt;
} //如果回家的话,左边没有和他相关的点。
//不用考虑这个点的出度
for (int i = 1;i <= n;i++){
int x;
cin >> x;
if (r[i]==0) {
l[i] = ++cnt;
students++;
continue;
}
if (x==0) {
l[i] = ++cnt;
students++;
}
} for (int i = 1;i <= n;i++)
for (int j = 1;j <= n;j++)
cin >> a[i][j]; for (int i = 1;i <= n;i++){
//是本校学生,但是回家
//不用给它安排
if (l[i]==0){
continue;
}
if (l[i]>0 && r[i]==0){
//不是本校学生
for (int j = 1;j <= n;j++)
if (i!=j && a[i][j] && r[j]!=0){//那个人必须是本校学生
add(l[i],r[j],1);
}
} //是本校学生,且不走
if (l[i]>0 && r[i]>0){
for (int j = 1;j <= n;j++)
if ( (a[i][j] && r[j]!=0) || (i==j)){
add(l[i],r[j],1);
}
}
} for (int i = 1;i <= n;i++)
if (l[i]!=0){//这个人不回家
add(0,l[i],1);
} ++cnt; for (int i = 1;i <= n;i++)
if (r[i]!=0){//不是外校的学生就有床位
add(r[i],cnt,1);
} s = 0;t = cnt;
int ans = 0;
while (bfs(s,t)){
for (int i = 0;i <= cnt;i++) tfir[i] = fir[i];
ans += dfs(s,t,1000);
}
if (ans==students){
cout<<"^_^"<<endl;
}else{
cout<<"T_T"<<endl;
}
}
return 0;
}

【BZOJ 1433】[ZJOI2009]假期的宿舍的更多相关文章

  1. bzoj 1433: [ZJOI2009]假期的宿舍 -- 最大流

    1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MB Description Input Output Sample Input ...

  2. bzoj 1433: [ZJOI2009]假期的宿舍

    1433: [ZJOI2009]假期的宿舍 Description Input Output Sample Input 1 3 1 1 0 0 1 0 0 1 1 1 0 0 1 0 0 Sample ...

  3. BZOJ 1433 [ZJOI2009]假期的宿舍(网络流)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1433 [题目大意] 在校学生有自己的床,外校的则没有,如果两个同学相互认识,则可以借用 ...

  4. bzoj 1433: [ZJOI2009]假期的宿舍【匈牙利算法】

    i能睡j床的连边(i,j),跑最大匹配或者最大流,然后看看人数能不能对上总数即可 #include<iostream> #include<cstdio> #include< ...

  5. 洛谷 2055 BZOJ 1433 [ZJOI2009]假期的宿舍

    [题解] 既然是一人对应一床,那么显然可以用二分图匹配来做.俩人认识的话,如果其中一个a是在校学生,另一个b不回家,b就可以向a的床连边(a,b当然也可以是同一个人). 然后如果最大匹配数大于等于需要 ...

  6. 1433: [ZJOI2009]假期的宿舍

    1433: [ZJOI2009]假期的宿舍 链接 分析: 直接网络流,看是否满足即可. S向每个有需要的人连边,有床的向T连边,认识的人之间互相连边. 代码: #include<cstdio&g ...

  7. 1433. [ZJOI2009]假期的宿舍【二分图】

    Description 学校放假了······有些同学回家了,而有些同学则有以前的好朋友来探访,那么住宿就是一个问题.比如A 和B都是学校的学生,A要回家,而C来看B,C与A不认识.我们假设每个人只能 ...

  8. BZOJ1433 ZJOI2009 假期的宿舍 二分图匹配

    1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2375  Solved: 1005[Submit][Sta ...

  9. bzoj1433: [ZJOI2009]假期的宿舍

    1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2286  Solved: 969[Submit][Stat ...

  10. bzoj1433:[ZJOI2009]假期的宿舍

    明显的二分图最大匹配. #include<cstdio> #include<cstring> #include<cctype> #include<algori ...

随机推荐

  1. @value 注解获取属性文件中的值

    一.属性文件 db.properties name=jack 二.配置文件 applicationContext.xml <!-- 加载配置文件,该节点只能存在一个,所以用 * ,加载所有属性文 ...

  2. BindingResult不能获取错误对象

    BindingResult不能获取错误对象,代码如下: @RequestMapping(value = "/login") public String error4( Model ...

  3. BA-siemens-insight_designer不支持win7 64位操作系统

    如果把insight安装在win7 64位操作系统中,在编辑图形界面的时候,designer会出现下面的错误对话框,侬还是老老实实的安装32位的系统吧!

  4. Hadoop Word Count程序

    Hadoop Word Count程序 pom.xml文件: <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns ...

  5. python的urlencode与urldecode

    ```python3.x中urlencode在urllib.parse模块中``` 当url地址含有中文,或者参数有中文的时候,这个算是很正常了,但是把这样的url作为参数传递的时候(最常见的call ...

  6. oracle实现查询每个部门的员工工资排在前三的员工的基本信息具体举例

    --先删除原先存在的表: drop table emp; --创建表emp create table emp ( deptno number, ename varchar2(20), sal numb ...

  7. HDU 1042.N!【高精度乘法】【8月24】

    N! Problem Description Given an integer N(0 ≤ N ≤ 10000), your task is to calculate N!   Input One N ...

  8. 终端安全工具 gartner 排名

    Reviews for Endpoint Detection and Response Solutions What is Endpoint Detection and Response Soluti ...

  9. USACO 2.2 Runaround Numbers

    Runaround Numbers Runaround numbers are integers with unique digits, none of which is zero (e.g., 81 ...

  10. Re:从 0 开始的微服务架构--(四)如何保障微服务架构下的数据一致性--转

    原文地址:http://mp.weixin.qq.com/s/eXvoJew3bjFKzLLJpS0Otg 随着微服务架构的推广,越来越多的公司采用微服务架构来构建自己的业务平台.就像前边的文章说的, ...