• UVa 1638 Pole Arrangement (递推或DP)


    题意:有高为1,2,3...n的杆子各一根排成一行,从左边能看到L根,从右边能看到R根,求杆子的排列有多少种可能。

    析:设d(i, j, k)表示高度为1-i的杆子排成一行,从左边看到j根,从右边看到k根的数目。当i>1时,我们按照从大到小的顺序按排杆子,

    假设已经安排完i-1根了,那么还剩下一根就是高度为1的了,那么它放在哪都不会挡住任何一根杆子。情况有3种:

    1.放在左边,那么在左边一定能够看到它,在右边看不到(因为i>1);

    2.放在右边,那么在右边一定能够看到它,在左边看不到(因为i>1);

    3.剩下的也就是放在中间了,那么我们有多少个地方可以放呢,答案很明显,i-2个,在左右两边都看不到它。

    情况1时,在高度在2-i的杆子中,在左边能看到j-1根,从右边能看到k根,因为在左边加上第一根正好是j根,

    同理,情况2,在高度在2-i的杆子中,在左边能看到j根,从右边能看到k-1根,因为在右边加上第一根正好是k根,

    情况3呢?更好说,在高度在2-i的杆子中,在左边能看到j根,从右边能看到k根,因为在左边和右边都看不到第一根。

    综上分析:d(i, j, k) = d(i-1, j-1, k) + d(i-1, j, k-1) + d(i-1, j, k) * (i-2),边界为d(1, 1 ,1) = 1;

    可先把所以情况先扫出来,到时直接查询就行了,快捷方便。注意超过int了,要用long long存储。

    代码如下:

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #define mod %10056
    
    using namespace std;
    typedef long long LL;
    LL d[25][25][25];
    
    void init(){
        memset(d, 0, sizeof(d));
        d[1][1][1] = 1;
        for(int i = 2; i <= 20; ++i)
            for(int j = 1; j <= 20; ++j)
                for(int k = 1; k <= 20; ++k)
                    d[i][j][k] = d[i-1][j-1][k] + d[i-1][j][k-1] + d[i-1][j][k] * (i-2);
    }
    
    int main(){
        init();
        int T, n, l, r;  cin >> T;
        while(T--){
            scanf("%d %d %d", &n, &l, &r);
            printf("%lld
    ", d[n][l][r]);
        }
        return 0;
    }
    
  • 相关阅读:
    多线程爬取斗图啦图片
    fiddler配置https
    Linux相关命令实例及解析
    htm、html、shtml网页区别
    什么是中间件?常见中间件有哪些?
    列举常见的关系型数据库和非关系型都有那些?
    什么是dao模式,dao模式的实现方法
    如何理解fine-grained和coarse-grained?
    .Net 理解持久层(Persistence Layer)
    web server与app server有什么不同
  • 原文地址:https://www.cnblogs.com/dwtfukgv/p/5537299.html
Copyright © 2020-2023  润新知