zoukankan      html  css  js  c++  java
  • 瞬间移动(组合数, 逆元)

    瞬间移动

    Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total Submission(s): 263    Accepted Submission(s): 143

    Problem Description
    有一个无限大的矩形,初始时你在左上角(即第一行第一列),每次你都可以选择一个右下方格子,并瞬移过去(如从下图中的红色格子能直接瞬移到蓝色格子),求到第n行第m列的格子有几种方案,答案对1000000007取模。
    ![http://acm.hdu.edu.cn/data/images/C702-1003-1.jpg](http://acm.hdu.edu.cn/data/images/C702-1003-1.jpg)
     
    Input
    多组测试数据。
    两个整数n,m(2n,m100000)
     
    Output
    一个整数表示答案
     
    Sample Input
    4 5
     
    Sample Output
    10
     
    Source

    题解:斜着看可以看出来其实是个杨辉三角,就是C(n + m - 4, m - 2);杨辉三角可以用逆元写;

    代码:

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<cmath>
    using namespace std;
    const int MOD = 1000000007;
    typedef __int64 LL;
    LL P[200010];
    void init(){
        P[0] = 1;
        for(int i = 1; i <= 200000; i++){
            P[i] = P[i - 1] * i % MOD;
        }
    }
    LL quick_mul(LL a, LL n){
        LL ans = 1;
        while(n){
            if(n & 1){
                ans *= a;
                ans %= MOD;
            }
            n >>= 1;
            a *= a;
            a %= MOD;
        }
        return ans % MOD;
    }
    LL C(int n, int m){
        LL a = P[n];
        LL b = P[m] * P[n - m]%MOD;
        LL p = quick_mul(b, MOD - 2);
        return a * p % MOD;
    }
    int main(){
        init();
        int n, m;
        while(~scanf("%d%d", &n, &m)){
            printf("%I64d
    ", C(n + m - 4, m - 2));
        }
        return 0;
    }

     还有两种必定超时的方法:

    #include<iostream>
    #include<cstdio>
    #include<cmath>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    const int MAXN = 1010;
    int dp[MAXN][MAXN];
    int main(){
        int n, m;
        while(~scanf("%d%d", &n, &m)){
            memset(dp, 0, sizeof(dp));
            for(int i = 2; i <= n; i++){
                for(int j = 2; j <= m; j++){
                    for(int p = 1; p < i; p++){
                        for(int k = 1; k < j; k++){
                            dp[i][j] += dp[p][k];
                        }
                    }
                }
            }
            printf("%d
    ", dp[n][m]);
        }
        return 0;
    }


    记忆化搜索;

    #include<iostream>
    #include<cstdio>
    #include<cmath>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    const int MAXN = 1010;
    int dp[MAXN][MAXN];
    int dfs(int x, int y){
        if(x <= 0 || y <= 0)
            return 0;
        if(dp[x][y])
            return dp[x][y];
        for(int i = 1; i < x; i++){
            for(int j = 1; j < y; j++){
                dp[x][y] += dfs(i, j);
            }
        }
        return dp[x][y];
    }
    int main(){
        int n, m;
        while(~scanf("%d%d", &n, &m)){
            dp[1][1] = 1;
            printf("%d
    ", dfs(n, m));
        }
        return 0;
    }
  • 相关阅读:
    C++拷贝构造函数具体解释
    兼容安卓的javaproject1.0
    php课程 12-40 抽象类的作用是什么
    php中类文件名的命名的规则是什么
    妙味css3课程---1-2、css3中新增的伪类和伪元素有哪些
    excel表如何实现多if选择结构多分支判断
    php如何读写excel
    php课程 12-39 继承中parent的作用是什么
    Dcloud课程9 天气小助手如何实现
    Dcloud课程8 开心一刻应用如何实现
  • 原文地址:https://www.cnblogs.com/handsomecui/p/5517838.html
Copyright © 2011-2022 走看看