YBTOJ&洛谷P3231:消毒(二分图匹配)

题目描述

最近在生物实验室工作的小 T 遇到了大麻烦。 由于实验室最近升级的缘故,他的分格实验皿是一个长方体,其尺寸为 a ∗ b ∗ c a*b*c abc。为了实验的方便,它被划分为 a ∗ b ∗ c a*b*c abc 个单位立方体区域,每个单位立方体尺寸为 1 ∗ 1 ∗ 1 1*1*1 111,并用 ( i , j , k ) (i,j,k) (i,j,k) 标识一个单位立方体。这个实验皿已经很久没有人用了。现在,小 T 被导师要求将其中一些单位立方体区域进行消毒操作(每个区域可以被重复消毒)。

而由于严格的实验要求,他被要求使用一种特定的 F 试剂来进行消毒。 这种 F 试剂特别奇怪,每次对尺寸为 x ∗ y ∗ z x*y*z xyz 的长方体区域(它由 x ∗ y ∗ z x*y*z xyz 个单位立方体组成)进行消毒时,只需要使用 m i n ( x , y , z ) min(x,y,z) min(x,y,z) 单位的 F 试剂。F 试剂的价格不菲,这可难倒了小 T。

现在请你告诉他,最少要用多少单位的 F 试剂。

解析

非暴力,不合作

首先可以有一个结论:**每次使 m i n ( x , y , z ) = 1 min(x,y,z)=1 min(x,y,z)=1,一定是不劣的
所以我们就每次一面一面的涂
看一个《弱化版》的题目

在一 N ∗ N N*N NN个 的矩阵中,有 K K K个格子中有杂物,现在你有一种能力,一次可以消除一行或一列格子中的杂物,问你至少需要几次可以将这些杂物全部消完。

这题应该是二分图的入门题了,把每个点的x坐标与y坐标相连,跑二分图最大匹配即可
不难发现,消毒这题应该就是消除杂物的升级版,从二维变成了三维
然鹅很快我们就发现并不能推广到k维。。。
当然如果您能发明三分图匹配本题就和喝水一样
那么我们怎么办呢?
然后就点开了题解
还是暴力的思想了
考虑到数据范围:
a ∗ b ∗ c < = 5000 a*b*c<=5000 abc<=5000

那么a、b、c中的最小值应该不超过17
所以我们考虑暴力枚举最小的一维的选取状态,然后每次跑一遍匈牙利取答案最小值即可

另外本题还有一个很巧妙的实现技巧
先把坐标存到三个一维数组里
把最小的一维swap到a的位置同时把对应的那一位的数组swap到第一位
代码实现就变得很简单了

代码

#include<bits/stdc++.h>
using namespace std;
const int N=5020;
#define ll long long
ll read(){
	ll x=0,f=1;char c=getchar();
	while(!isdigit(c)){if(c=='-')f=-1;c=getchar();};
	while(isdigit(c)){x=x*10+c-'0';c=getchar();};
	return x*f;
}
int n,m,l;
struct node{
	int from,to,nxt;
}p[N];
int fi[N],cnt=-1;
void addline(int x,int y){
	p[++cnt]=(node){x,y,fi[x]};
	fi[x]=cnt;
}
int vis[N],mat[N];
bool dfs(int x,int tim){
	if(vis[x]==tim) return false;
	vis[x]=tim;
	for(int i=fi[x];~i;i=p[i].nxt){
		int to=p[i].to;
		if(!mat[to]||dfs(mat[to],tim)){
			mat[to]=x;return true;
		}
	}
	return false;
}
int a,b,c;
int hungary(){
	int res=0;
	memset(vis,0,sizeof(vis));
	memset(mat,0,sizeof(mat));
	for(int i=1;i<=b;i++){
		if(dfs(i,i)){
			res++;//printf("  ok:%d\n",i);
		}
	}
	return res;
}


int q[4][N],num;
bool ok[35];
int ans;
int calc(){
	memset(fi,-1,sizeof(fi));cnt=-1;
	for(int i=1;i<=num;i++){
		if(ok[q[1][i]]) continue;
		int x=q[2][i],y=q[3][i];
		addline(x,y+b);addline(y+b,x);
	//	printf("x=%d y=%d\n",x,y);
	}
	return hungary();
}
void find(int k,int val){
	if(k>a){//printf("ok");
		//printf("-------------val=%d\n",val);
		//for(int i=1;i<=a;i++) printf("%d ",ok[i]);
		//printf("\n");
		ans=min(ans,calc()+val);
		//printf("---ans=%d\n\n",ans);
		return;
	}
	find(k+1,val);
	ok[k]=1;find(k+1,val+1);
	ok[k]=0;
}
int main(){
	int T=read();
	while(T--){
		a=read();b=read();c=read();num=0;ans=5000;
		int mn=min(a,min(b,c));
		for(int i=1;i<=a;i++){
			for(int j=1;j<=b;j++){
				for(int k=1;k<=c;k++){
					int x=read();
					if(!x) continue;
					q[1][++num]=i;q[2][num]=j;q[3][num]=k;
				}
			}
		}
		if(mn==b){
			swap(a,b);swap(q[1],q[2]);
		}
		else if(mn==c){
			swap(a,c);swap(q[1],q[3]);
		}
		find(1,0);
		printf("%d\n",ans);
	}
	return 0;
}
/*

*/
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值