zoukankan      html  css  js  c++  java
  • 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;
    }
    
  • 相关阅读:
    [android] 实现返回键操作思路
    [android] 切换界面的问题
    [android] 切换界面的通用处理
    [android] 界面切换的简单动画
    [android] 界面切换的核心方法
    [android] 标题部分管理
    [android] 界面的划分
    [android] socket在手机上的应用
    [android] 网络链接类型和渠道
    [android] android通信协议
  • 原文地址:https://www.cnblogs.com/dwtfukgv/p/5537299.html
Copyright © 2011-2022 走看看