Hash_P1026毒药?解药?

时间:2023-01-12 17:45:54
 #include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
int n,m,map[][],dist[][],hash[],st[];
void read(int &x){
x=; int f=; char ch;
for (ch=getchar();!isdigit(ch);ch=getchar()) if (ch=='-') f=-;
for (;isdigit(ch);ch=getchar()) x=x*+ch-''; x*=f;
}
int Hash(int *a){
int x=;
for (int i=;i<=n;i++){
x<<=;
if (a[i]==) x++;
}return x;
}
void bfs(){
int head=,tail=,goal,x;
goal=(<<n)-; hash[]=;
dist[++tail][]=;
while (head<tail){
head++;
for (int i=;i<=m;i++){
for (int j=;j<=n;j++){
if (map[i][j]==) st[j]=;
else if (map[i][j]==-) st[j]=;
else st[j]=dist[head][j];
}
x=Hash(st);
if (!hash[x]){
hash[x]^=; tail++;
dist[tail][]=dist[head][]+;
for (int j=;j<=n;j++) dist[tail][j]=st[j];
}
if (hash[goal]==){
printf("%d\n",dist[tail][]);
return;
}
}
}
printf("The patient will be dead.\n");
}
int main(){
memset(hash,,sizeof(hash));
read(n),read(m);
for (int i=;i<=m;i++)
for (int j=;j<=n;j++)
read(map[i][j]);
bfs();
return ;
}

题目链接:https://vijos.org/p/1026。

题目大意:羽毛笔和im是抽签到同一个考场的,她们突然闻到一阵刺鼻的化学试剂的气味。

机灵鼠:(头都不抬)你们是考生么?还在门口磨蹭什么?快进来帮我忙!!……怎么还不进来?你们拖赛,拖赛,把你们的青春都拖掉赛……
im:开…开策了>_<
羽毛笔:哎呀~~机灵鼠大人要我们帮什么忙?^^
机灵鼠:你们看这里的这些药,都是我研制的对付各种症状的解药。可是我一个不小心,每种药都小小地配错了一点原料,所以这些药都有可能在治愈某些病症的同时又使人患上某些别的病症……(im:那…那是解药还是毒药啊?!)……经过我天才的努力(背景:我是天才!!),终于弄清了每种药的具体性能(路人甲:那是你自己配的吗?-_-),我会把每种药能治的病症和能使人患上的病症列一张清单给你们,然后你们要根据这张清单找出能治愈所有病症的最少药剂组合……顺便说一声,病症的数目不超过10种(小呆:偶是好人吧^^),我的药是用不完的,就是说每种药剂都可以被重复使用。给你们的单子里第一行是病症的总数n,第二行是药剂的种类m(0<m<=100),以下有m行,每行有n个数字用空格隔开,文件的第i+2行的n个数字中,如果第j个数为1,就表示第i种药可以治愈病症j(如果患有这种病的话则治愈,没有这种病则无影响),如果为0表示无影响,如果为-1表示反而能使人得上这种病(无病患上,有病无影响)。我制的药任何两种性能都不同。你们只要给我用的最少的药剂数就可以了。

做法:数据范围比较小,我们考虑搜索,求最少的药剂数,我们可以用BFS,如果我们直接暴力的话会超时,我们发现只有10中病症状态数只有1024级别,对于相同的状态我们显然不用重复入队,这一过程我们用hash来判重即可。

bfs+Hash。