1066: [SCOI2007]蜥蜴
Description
在一个r行c列的网格地图中有一些高度不同的石柱,一些石柱上站着一些蜥蜴,你的任务是让尽量多的蜥蜴逃
到边界外。 每行每列中相邻石柱的距离为1,蜥蜴的跳跃距离是d,即蜥蜴可以跳到平面距离不超过d的任何一个石
柱上。石柱都不稳定,每次当蜥蜴跳跃时,所离开的石柱高度减1(如果仍然落在地图内部,则到达的石柱高度不
变),如果该石柱原来高度为1,则蜥蜴离开后消失。以后其他蜥蜴不能落脚。任何时刻不能有两只蜥蜴在同一个
石柱上。
Input
输入第一行为三个整数r,c,d,即地图的规模与最大跳跃距离。以下r行为石竹的初始状态,0表示没有石柱
,1~3表示石柱的初始高度。以下r行为蜥蜴位置,“L”表示蜥蜴,“.”表示没有蜥蜴。
Output
输出仅一行,包含一个整数,即无法逃离的蜥蜴总数的最小值。
Sample Input
5 8 2
00000000
02000000
00321100
02000000
00000000
........
........
..LLLL..
........
........
00000000
02000000
00321100
02000000
00000000
........
........
..LLLL..
........
........
Sample Output
1
HINT
100%的数据满足:1<=r, c<=20, 1<=d<=4
问题转化成最多有多少只蜥蜴可以逃出去。。就变成了最大流。。。石柱的高度只需,把石柱拆成两个点,中间连一条流量为高度的边。。。然后建图即可。。。
1 #include<iostream> 2 #include<cstdlib> 3 #include<cmath> 4 #include<cstring> 5 #include<cstdio> 6 #include<algorithm> 7 #include<string> 8 #include<map> 9 #include<queue> 10 #include<vector> 11 #include<set> 12 #define inf 1000000000 13 #define maxn 805 14 #define maxm 500000+5 15 #define eps 1e-10 16 #define ll long long 17 #define for0(i,n) for(int i=0;i<=(n);i++) 18 #define for1(i,n) for(int i=1;i<=(n);i++) 19 #define for2(i,x,y) for(int i=(x);i<=(y);i++) 20 #define for3(i,x,y) for(int i=(x);i>=(y);i--) 21 #define for4(i,x) for(int i=head[x],y=e[i].go;i;i=e[i].next,y=e[i].go) 22 using namespace std; 23 int read(){ 24 int x=0,f=1;char ch=getchar(); 25 while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} 26 while(ch>='0'&&ch<='9'){x=10*x+ch-'0';ch=getchar();} 27 return x*f; 28 } 29 int n,m,s,t,maxflow,tot=1,head[maxn],cur[maxn],h[maxn],cnt,sum; 30 int mp[25][25],mp2[25][25],id[25][25],r,c,d; 31 queue<int>q; 32 struct edge{int go,next,v;}e[maxm]; 33 bool check(int x1,int y1,int x2,int y2){ 34 if((x1-x2)*(x1-x2)+(y1-y2)*(y1-y2)<=d*d)return 1; 35 return 0; 36 } 37 void insert(int x,int y,int v) 38 { 39 e[++tot]=(edge){y,head[x],v};head[x]=tot; 40 e[++tot]=(edge){x,head[y],0};head[y]=tot; 41 } 42 bool bfs() 43 { 44 for(int i=s;i<=t;i++)h[i]=-1; 45 q.push(s);h[s]=0; 46 while(!q.empty()) 47 { 48 int x=q.front();q.pop(); 49 for(int i=head[x];i;i=e[i].next) 50 if(e[i].v&&h[e[i].go]==-1) 51 { 52 h[e[i].go]=h[x]+1;q.push(e[i].go); 53 } 54 } 55 return h[t]!=-1; 56 } 57 int dfs(int x,int f) 58 { 59 if(x==t) return f; 60 int tmp,used=0; 61 for(int i=cur[x];i;i=e[i].next) 62 if(e[i].v&&h[e[i].go]==h[x]+1) 63 { 64 tmp=dfs(e[i].go,min(e[i].v,f-used)); 65 e[i].v-=tmp;if(e[i].v)cur[x]=i; 66 e[i^1].v+=tmp;used+=tmp; 67 if(used==f)return f; 68 } 69 if(!used) h[x]=-1; 70 return used; 71 } 72 void dinic() 73 { 74 maxflow=0; 75 while(bfs()) 76 { 77 for (int i=s;i<=t;i++)cur[i]=head[i];maxflow+=dfs(s,inf); 78 } 79 } 80 81 int main(){ 82 //freopen("input.txt","r",stdin); 83 //freopen("output.txt","w",stdout); 84 r=read();c=read();d=read();s=0;t=801; 85 for1(i,r) 86 for1(j,c){ 87 sum++;id[i][j]=sum; 88 char ch=getchar(); 89 while(ch<'0'||ch>'9')ch=getchar(); 90 mp[i][j]=ch-'0'; 91 } 92 for1(i,r) 93 for1(j,c){ 94 char ch=getchar(); 95 while(ch!='.'&&ch!='L')ch=getchar(); 96 if(ch=='L'){insert(0,id[i][j],1);cnt++;} 97 } 98 for1(i,r) 99 for1(j,d){ 100 insert(id[i][j]+400,801,inf); 101 insert(id[i][c-j+1]+400,801,inf); 102 } 103 for1(i,c) 104 for1(j,d){ 105 insert(id[j][i]+400,801,inf); 106 insert(id[r-j+1][i]+400,801,inf); 107 } 108 for(int x1=1;x1<=r;x1++) 109 for(int y1=1;y1<=c;y1++) 110 for(int x2=x1-d;x2<=x1+d;x2++) 111 for(int y2=y1-d;y2<=y1+d;y2++) 112 if(check(x1,y1,x2,y2)&&(x1!=x2||y1!=y2))insert(id[x1][y1]+400,id[x2][y2],inf); 113 for1(i,r) 114 for1(j,c) 115 if(mp[i][j])insert(id[i][j],id[i][j]+400,mp[i][j]); 116 dinic(); 117 printf("%d",cnt-maxflow); 118 return 0; 119 }