zoukankan      html  css  js  c++  java
  • 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
    ",gs());
        return;
    }
    int main(){
        int t;
        scanf("%d",&t);
        while(t--)solve();
        return 0;
    }
  • 相关阅读:
    8.20Java之反射机制的基本概念
    8.18Go语言之字符串
    Debug
    Feign
    Nacos
    SpringCloud Alibaba
    SpringCloud
    Maven
    Maven
    Jenkins
  • 原文地址:https://www.cnblogs.com/yzxverygood/p/11413315.html
Copyright © 2011-2022 走看看