CF 348 D. Turtles

D. Turtles

链接

题意:

  给定一个N*M的棋盘,有些格子不能走,问有多少种从(1,1)到(N,M)的两条不相交路径。

分析:

  lGV定理

  定理:点集A={a1,a2,…an}A={a1,a2,…an}到B={b1,b2,…bn}B={b1,b2,…bn}的不相交路径条数等于下面矩阵的行列式。

  [e(a1,b1)e(a1,b2)e(a1,bn)e(a2,b1)e(a2,b2)e(a2,bn)e(an,b1)e(an,b2)e(an,bn)]

  e(a,b)为从点a到点b的路径条数,本质是容斥。

  这道题目中,任意一条合法的路径都是从(1,2)->(n-1,m)和(2,1)->(n,m-1)的,所以A={(1,2),(2,1)}B={(n1,m),(n,m1)}。  

代码:

复制代码
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<cmath>
#include<cctype>
#include<set>
#include<queue>
#include<vector>
#include<map>
using namespace std;
typedef long long LL;

inline int read() {
    int x=0,f=1;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-1;
    for(;isdigit(ch);ch=getchar())x=x*10+ch-'0';return x*f;
}

const int N = 3005, mod = 1e9 + 7;
int f[N][N];
char s[N][N];

int Calc(int a,int b,int c,int d) {
    memset(f, 0, sizeof(f));
    for (int i = a; i <= c; ++i) 
        for (int j = b; j <= d; ++j) 
            if (s[i][j] == '.') {
                if (i == a && j == b) f[i][j] = 1;
                else f[i][j] = (f[i - 1][j]  + f[i][j - 1]) % mod;
            }
    return f[c][d];
}
int main() {
    int n = read(), m = read();
    for (int i = 1; i <= n; ++i) scanf("%s", s[i] + 1); 
    LL t1 = Calc(1, 2, n - 1, m), t2 = Calc(2, 1, n, m - 1);
    LL t3 = Calc(1, 2, n, m - 1), t4 = Calc(2, 1, n - 1, m);
    cout << (t1 * t2 % mod - t3 * t4 % mod + mod) % mod;
    return 0;
}
复制代码

 

posted @   MJT12044  阅读(222)  评论(0编辑  收藏  举报
编辑推荐:
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
· Linux系统下SQL Server数据库镜像配置全流程详解
阅读排行:
· Sdcb Chats 技术博客:数据库 ID 选型的曲折之路 - 从 Guid 到自增 ID,再到
· 语音处理 开源项目 EchoSharp
· 《HelloGitHub》第 106 期
· mysql8.0无备份通过idb文件恢复数据过程、idb文件修复和tablespace id不一致处
· 使用 Dify + LLM 构建精确任务处理应用
点击右上角即可分享
微信分享提示