hdu1045 DFS

时间:2023-03-08 16:53:11
hdu1045 DFS
#include<stdio.h>
#include<string.h>
int n;
int maxx;
char map[][];
int dx[]={,-,,};
int dy[]={,,,-}; int block[][][];//炮台位置 bool ok(int x,int y){
int i,x1,y1;
if(map[x][y]!='.')
return false;
for(i=;i<;i++){
x1=x+dx[i];
y1=y+dy[i];
while(){
if(x1<||x1>=n||y1<||y1>=n||map[x1][y1]=='X')//遇到边界跳出来
break;
else if(map[x1][y1]=='')//遇到'X'跳出来
return false;
x1+=dx[i];
y1+=dy[i];//没有的话就沿着行和列一直找
}
}
return true;
} void dfs(int k){
int i,j;
for(i=;i<n;i++){
for(j=;j<n;j++){
if(ok(i,j)){//判断是否能放置
map[i][j]='';//如果能将其所在行和列标记为'1',不能放置直到碰到'X'
dfs(k+);//成功放置的话就加1
map[i][j]='.';//回溯
}
}
if(maxx<k){//寻找最大数量
maxx=k;
}
}
} int main(){
int i,j;
while(scanf("%d",&n)!=EOF&&n){
maxx=;
for(i=;i<n;i++){
scanf("%s",map[i]);
}
dfs();
printf("%d\n",maxx);
}
return ;
}

二分图方法:

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
int n1,n2;
char map[][]; //数组开大点
int mapx[][],mapy[][];
int ma[][];//邻接矩阵true代表有边相连
int result[],visit[];
int x,y; int find(int a){
int i;
for(i=;i<=y;i++){
if(!visit[i]&&ma[a][i]){//如果节点i与a相邻并且未被查找过
visit[i]=;//标记i为已查找过
if(!result[i]||find(result[i])){//如果i未在前一个匹配M中或者i在匹配M中,但是从与i相邻的节点出发可以有增广路
result[i]=a;//记录查找成功记录
return ;
}
}
}
return ;
} int main(){
int i,j,ans;
while(scanf("%d",&n1)!=EOF&&n1){
n2=n1;
for(i=;i<n1;i++){
scanf("%s",map[i]);
}
memset(mapx,,sizeof(mapx));
memset(mapy,,sizeof(mapy));
x=;
for(i=;i<n1;i++){
for(j=;j<n2;j++){
if(map[i][j]=='.'){
++x;
while(j<n2&&map[i][j]=='.'){
mapx[i][j]=x;
j++;
}
}
}
}
y=;
for(j=;j<n2;j++){
for(i=;i<n1;i++){
if(map[i][j]=='.'){
++y;
while(i<n1&&map[i][j]=='.'){
mapy[i][j]=y;
i++;
}
}
}
} for(i=;i<n1;i++){
for(j=;j<n2;j++){
ma[mapx[i][j]][mapy[i][j]]=;
}
} ans=; memset(result,,sizeof(result));
for(i=;i<=x;i++){
memset(visit,,sizeof(visit));//清空上次搜索时的标记
ans+=find(i); //从节点i尝试扩展
}
printf("%d\n",ans);
}
return ;
}