GCJ 2015R2(Pegman-试出来)

Problem A. Pegman

Problem

While using Google Street View, you may have picked up and dropped the character Pegman before. Today, a mischievous user is going to place Pegman in some cell of a rectangular grid of unit cells with R rows and C columns. Each of the cells in this grid might be blank, or it might be labeled with an arrow pointing in one of four possible directions: up, right, down, or left.

When Pegman is placed on a grid cell, if that cell is blank, Pegman stands still forever. However, if that cell has an arrow, Pegman starts to walk in that direction. As he walks, whenever he encounters a blank cell, he just keeps walking in his current direction, but whenever he encounters another arrow, he changes to the direction of that arrow and then keeps walking.

You know that it is possible that Pegman might keep happily walking around and around the grid forever, but it is also possible that Pegman's walk will take him over the edge of the grid! You may be able to prevent this and save him by changing the direction of one or more arrows. (Each arrow's direction can only be changed to one of the other three possible directions; arrows can only be changed, not added or removed.)

What is the smallest number of arrows you will need to change to ensure that Pegman will not walk off the edge, no matter where on the grid he is initially placed?

Input

The first line of the input gives the number of test cases, TT test cases follow. Each begins with one line with two space-separated integers RC. This line is followed by Rlines, each of which has C characters, each of which describes a grid cell and is one of the following:

. period = no arrow
^ caret = up arrow
> greater than = right arrow
v lowercase v = down arrow
< less than = left arrow

Output

For each test case, output one line containing "Case #x: y", where x is the test case number (starting from 1) and y is the minimum number of arrows that must be changed to ensure that Pegman will not leave the grid no matter where he is initially placed, or the text IMPOSSIBLE if it is not possible to ensure this, no matter how many arrows you change.

Limits

1 ≤ T ≤ 100.

Small dataset

1 ≤ R, C ≤ 4.

Large dataset

1 ≤ R, C ≤ 100.

Sample


Input 
 

Output 
 
4
2 1
^
^
2 2
>v
^<
3 3
...
.^.
...
1 1
.

Case #1: 1
Case #2: 0
Case #3: IMPOSSIBLE
Case #4: 0
In Case #1, Pegman is guaranteed to walk off the top edge of the grid, no matter where he is placed. You can prevent that by changing the topmost arrow to point down, which will cause him to walk back and forth between those two arrows forever.

In Case #2, no matter where Pegman is placed, he will walk around and around the board clockwise in a circle. No arrows need to be changed.

In Case #3, the mischievous user might place Pegman on the up arrow in the middle of the grid, in which case he will start walking and then walk off the top edge of the grid. Changing the direction of this arrow won't help: it would just make him walk off a different edge.

In Case #4, the only possible starting cell is blank, so Pegman will stand still forever and is in no danger.



试玩一下,发现:

1.某个箭头独占一行与一列无解,否则肯定有解(互相指,或指向有解那个)

2.若有解,某条既定无解路,只需要改变最后1个点就行了

3.没更优方案




#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<functional>
#include<iostream>
#include<cmath>
#include<cctype>
#include<ctime>
#include<map>
#include<set>
#include<queue>
#include<stack>
#include<queue>
#include<vector>
#include<deque>
#include<complex>
using namespace std;
#define For(i,n) for(int i=1;i<=n;i++)
#define Fork(i,k,n) for(int i=k;i<=n;i++)
#define Rep(i,n) for(int i=0;i<n;i++)
#define ForD(i,n) for(int i=n;i;i--)
#define RepD(i,n) for(int i=n;i>=0;i--)
#define Forp(x) for(int p=pre[x];p;p=next[p])
#define Forpiter(x) for(int &p=iter[x];p;p=next[p])  
#define Lson (x<<1)
#define Rson ((x<<1)+1)
#define MEM(a) memset(a,0,sizeof(a));
#define MEMI(a) memset(a,127,sizeof(a));
#define MEMi(a) memset(a,128,sizeof(a));
#define INF (2139062143)
#define F (100000007)
#define MAXN (100+10)
#define MAXT (100+10)
typedef long long ll;
ll mul(ll a,ll b){return (a*b)%F;}
ll add(ll a,ll b){return (a+b)%F;}
ll sub(ll a,ll b){return (a-b+(a-b)/F*F+F)%F;}
void upd(ll &a,ll b){a=(a%F+b%F)%F;}
char c[MAXN][MAXN];
int a[MAXN][MAXN];
bool b[MAXN][MAXN];
int n,m;
int h[MAXN];
bool is_ok()
{
	For(i,n)
	{
		For(j,m)
			if (a[i][j])
			{
				int p=0;
				For(k,m) if(a[i][k]) ++p;
				For(k,n) if(a[k][j]) ++p;
				if (p==2) return 0;
			}
	}
	return 1;
		
}
bool insi(int i,int j)
{
	return 1<=i&&i<=n&&1<=j&&j<=m;
}
int dir[5][3]={{},{0,-1,0},{0,1,0},{0,0,-1},{0,0,1}}; 
int dfs(int i,int j,int di,int fi,int fj)
{
	if (!(i==fi&&j==fj))
	{
		if (b[i][j]) return 0;
		
		if (!insi(i,j)) return 1;
	}
	if (a[i][j]) b[i][j]=1,di=a[i][j];	
	
	dfs(i+dir[di][1],j+dir[di][2],di,fi,fj);
}
int main()
{
	freopen("A-large.in","r",stdin);
//	freopen("A-large.out","w",stdout);
	
	h['.']=0,h['^']=1,h['v']=2,h['<']=3,h['>']=4;
	
	int T;
	cin>>T;
	For(kcase,T)
	{
		cin>>n>>m;
		MEM(b)
		For(i,n) scanf("%s",c[i]+1);
		
		For(i,n) 
		{
			For(j,m)
				a[i][j]=h[c[i][j]];
		}
		
		if (!is_ok())
		{
			printf("Case #%d: IMPOSSIBLE\n",kcase);
			continue;
		}
		
		ll ans=0;

		For(i,n)
			For(j,m)
				if (a[i][j]&&!b[i][j])
				{
					ans+=dfs(i,j,a[i][j],i,j);
					
				} 



		printf("Case #%d: %lld\n",kcase,ans);
	}

	
	return 0;
}





你可能感兴趣的:(GCJ 2015R2(Pegman-试出来))