洛谷 p2055 假期的宿舍 题解
好长时间没更博客了
因为实在太蒻了
这让本蒟蒻怎么办
今天终于遇到了一道模板题(之前也有,不过太蒻了都不会)
不过...写代码5分钟,调试2小时
分界线:回归正题
这个就是普通的匈牙利算法 差不多
思路:
我们需要统计出谁需要床,谁有床
我们的二分图就是 需要的人 和 床
跑匈牙利就好了
什么人不需要床?? 在校且回家的人
什么床能用?? 在校的人
由于人的序号和床的序号会重复
所以我在床的序号上加了m,(比如一共有3个人,一号床的"名字"就是4)
如果还不明白上面那句那就可以理解为#define 一号床 4
由于题目要求输入是邻接矩阵,所以我们可以只读入一半(左上角和右下角连线以左不读)
因为人的信任是相互的
对应好之后就可以了
做这道题,怎么做?
1 : 学会匈牙利算法,不管你用Google.Baidu.360.搜狗.bing.yandex......什么引擎(能用就行)
都有很生动的例子,这里不再赘述
2 : 学会图论基本知识,会用邻接表存边(只会邻接矩阵的话可以先学一学或者看懂我的之后自己写)
3 : 写代码,就是用你的小爪爪摸一下键盘
4 : debug,dalao自行跳过
5 : 最重要的一步:
打上神兽!!!!
┌─┐ ┌─┐
┌─┘ ┴────┘ ┴─┐
│ │
│ ─── │
│ ─┬┘ └┬─ │
│ │
│ ─┴─ │
│ │
└──┐ ┌──┘
│ │
│ │
│ │
│ │
│ │
│ │
│ │
│ └──────────┐
│ │
│ ├─┐
│ ┌─┘
│ │
└─┐ ┐ ┌────┬ ┐ ┌──┘
│─┤─┤ │─┤─┤
└─┴─┘ └─┴─┘
神兽保佑
代码无BUG!
看起来好像不大对,但是粘贴到记事本或sublime(Dev也行)
就会变成非常帅气的样子
现在贴代码
#include<bits/stdc++.h>
using namespace std;
inline int read(){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();return s*w;}
inline void write(int x) {if(x<0) putchar('-'),x=-x; if(x>9) write(x/10); putchar(x%10+'0'); }
int u[10001],v[10001],f[10001],N[10001],g[10001];
bool bj[10001],bj2[10001],book[10001];
void add(int x,int y,int z){//邻接表存边
u[z] = x; v[z] = y;
N[z] = f[x]; f[x] = z;
}
bool find(int x){//匈牙利模板
for(int i = f[x];i != -1;i = N[i]){
int p = v[i];
if(!book[p]){
book[p] = 1;
if(g[p] == 0 || find(g[p])){
g[p] = x;
return 1;
}
}
}
return 0;
}
int main(int argc, char const *argv[])
{
int yy;
yy = read();
while(yy){
int tt = 0;
yy--;
memset(f,-1,sizeof f);//多测不清空,爆零两行泪
memset(u,0,sizeof u);
memset(v,0,sizeof v);
memset(g,0,sizeof g);
memset(bj,0,sizeof bj);
memset(bj2,0,sizeof bj2);
int m,x,tot = 0;
m = read();
for(int i = 1;i <= m;++i) bj[i] = read();//在校标记
for(int i = 1;i <= m;++i) {
x = read();
bj2[i] = x;//是否回家的标记
if(!bj[i]) bj2[i] = 0;//不在校就可以当不回家处理(因为需要床)
if(bj2[i]) tt++;
}
for(int i = 1;i <= m;++i){
for(int j = 1;j <= m;++j){
x = read();
if(i > j) continue;//只读一半
if(i == j && bj[i])//自己可以睡自己的床
x = 1;
if(x == 0) continue;//不能互相睡床不存边
if(bj[j] && !bj2[i]){//j有床i不回家就加边
// cout<<i<<" "<<j<<endl;
add(i,m + j,++tot);
}
if(i == j) continue;
if(bj[i] && !bj2[j]){//同理
// cout<<j<<" "<<i<<endl;
add(j,m + i,++tot);
}
}
}
// for(int i = 1;i <= m;++i){
// cout<<f[i]<<" "<<N[i]<<endl;
// }
int cnt = 0;
for(int i = 1;i <= m;++i){
memset(book,0,sizeof book);
if(find(i)) cnt++;
}
if(cnt == m - tt) cout<<"^_^"<<"\n";
else cout<<"T_T"<<"\n";
}
return 0;
}
/***
* ┌─┐ ┌─┐
* ┌─┘ ┴────┘ ┴─┐
* │ │
* │ ─── │
* │ ─┬┘ └┬─ │
* │ │
* │ ─┴─ │
* │ │
* └──┐ ┌──┘
* │ │
* │ │
* │ │
* │ │
* │ │
* │ │
* │ │
* │ └──────────┐
* │ │
* │ ├─┐
* │ ┌─┘
* │ │
* └─┐ ┐ ┌────┬ ┐ ┌──┘
* │─┤─┤ │─┤─┤
* └─┴─┘ └─┴─┘
* 神兽保佑
* 代码无BUG!
*/
作者: liuzitong
出处:http://www.cnblogs.com/lztzs/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。