loj6259「CodePlus 2017 12 月赛」白金元首与独舞

分析

我们将没连的点连向周围四个点

其余的按照给定的方向连

我们将所有连出去的位置统一连到0点上

再以0作为树根

于是就将问题转化为了有向图内向树计数

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<algorithm>
#include<cctype>
#include<cmath>
#include<cstdlib>
#include<queue>
#include<ctime>
#include<vector>
#include<set>
#include<map>
#include<stack>
using namespace std;
const int mod = 1e9+7;
const int dx[] = {1,-1,0,0};
const int dy[] = {0,0,1,-1};
int n,m,g[1100][1100],id[1100][1100],cnt,ok;
char s[1100][1100];
inline void add(int x,int y){
    g[y][y]++;
    g[y][x]--;
}
inline int dfs(int x, int y, int len) {
    if(x<1||x>n||y<1||y>m)return 0;
    if(id[x][y]!=-1)return id[x][y];
    if(len>n*m){
      ok=1;
      return 0;
    }
    if(s[x][y]=='L')return id[x][y]=dfs(x,y-1,len+1);
    if(s[x][y]=='R')return id[x][y]=dfs(x,y+1,len+1);
    if(s[x][y]=='U')return id[x][y]=dfs(x-1,y,len+1);
    if(s[x][y]=='D')return id[x][y]=dfs(x+1,y,len+1);
}
inline int gs(){
    int i,j,k,ans=1;
    for(i=1;i<=cnt;i++)
      for(j=1;j<=cnt;j++)
        g[i][j]=(g[i][j]%mod+mod)%mod;
    for(i=1;i<=cnt;i++){
      for(j=i;j<=cnt;j++)
        if(g[i][j])break;
      if(j>cnt)return 0;
      if(j!=i)ans=mod-ans,swap(g[i],g[j]);
      for(j=i+1;j<=cnt;j++){
          while(g[j][i]){
            int t=g[i][i]/g[j][i];
            for(k=i;k<=cnt;k++)
              g[i][k]=(g[i][k]-1ll*t*g[j][k]%mod+mod)%mod;
            ans=mod-ans;
            swap(g[i],g[j]);
          }
      }
      ans=1ll*ans*g[i][i]%mod;
    }
    return ans;
}
inline void solve(){
    int i,j,k;
    ok=0;
    cnt=0;
    memset(g,0,sizeof(g));
    memset(id,-1,sizeof(id));
    scanf("%d%d",&n,&m);
    for(i=1;i<=n;i++)scanf("%s",s[i]+1);
    for(i=1;i<=n;i++)
      for(j=1;j<=m;j++)
        if(s[i][j]=='.')
          id[i][j]=++cnt;
    for(i=1;i<=n;i++)
      for(j=1;j<=m;j++)
        if(s[i][j]=='.'){
          for(k=0;k<4;k++)
            add(dfs(i+dx[k],j+dy[k],0),id[i][j]);
        }else dfs(i,j,0);
    if(ok){
      puts("0");
      return;
    }
    printf("%d\n",gs());
    return;
}
int main(){
    int t;
    scanf("%d",&t);
    while(t--)solve();
    return 0;
}
posted @ 2019-08-26 16:26  水题收割者  阅读(292)  评论(0编辑  收藏  举报