$SP15637\ GNYR04H\ -\ Mr\ Youngs\ Picture\ Permutations$
Description
杨先生希望为他的班级拍照。学生将排成一行,每行不超过后面的行,并且行的左端对齐。例如,可以安排12名学生排列(从后到前)5,3,3和1名学生。
X X X X X
X X X
X X X
X
此外,杨先生希望每排学生安排高度从左到右减少。此外,学生身高应从后向前减少。想想看,杨先生看到,对于这个12人的例子,至少有两种安排学生的方式(数字代表高度,其中1代表最高):
1 2 3 4 5 1 5 8 11 12
6 7 8 2 6 9
9 10 11 3 7 10
12 4
杨先生想知道,对于给定排列的排列,可能有多少不同的学生安排。他尝试用长度为3,2和1的行开始计数,并计数16个排列:
123 123 124 124 125 125 126 126 134 134 135 135 136 136 145 146
45 46 35 36 34 36 34 35 25 26 24 26 24 25 26 25
6 5 6 5 6 4 5 4 6 5 6 4 5 4 3 3
杨先生认为,手动点数对于任何合理数量的学生来说都不会很有效。他通过编写计算机程序来帮助你确定一组给定行的学生的不同安排数量。
Input
输入描述了一系列测试,每个测试分两行描述。第一行将行数k作为十进制整数。第二行包含从后到前的行的长度(n {1} 1 ,n {2} 2 ,...,n _ {K} ķ )作为由单个空格分隔的十进制整数。问题集以行计数为0的行结束。最多不会超过5行,学生总数N(行长度总和)最多不超过30行。
Output
对于每个测试用例输出一个整数:N个学生排列在给定行中的数量,以便高度从左到右沿着每行减少,并且从后到前沿着每列减小(假定所有高度都不同)。结果应该分开。输入数据将被选择,以便结果总是适合一个无符号的32位整数
Solution
大概就是排一个高最大为5的类似倒三角形,然后每一行的数字从左往右递增,每一列的数字从上往下递增,其实就是保证每一个数都大于它左边的数和它上边的数,求满足要求的排列方案数
第一反应就是最多5排啊,这么小就可以直接DP了啊:
$f[a][b][c][d][e]$表示第一排已经站了$a$个人,第二排站了$b$个人.....的方案数
转移:
从小到大的填,一个数可以填在一个位置当且仅当它的左边和上面的数已经填了,或者它就是最左边或最上面的位置,这样就可以保证它大于左边和上面的数了,还要注意每一行的人数不要超过输入给的人数
对了还要注意$MLE$的问题,开空间不能开$f[31][31][31][31][31]$,因为它是一个类似倒三角形(每一行的人数都小于或等于上一行的人数),所以可以推出只要开$f[31][31/2+1][31/3+1][31/4+1][31/5+1]$就可以了
Code
#include<iostream>
#include<cstdio>
#include<cstring>
#define Rg register
#define il inline
#define db double
#define ll unsigned long long
#define go(i,a,b) for(Rg int i=a;i<=b;i++)
#define yes(i,a,b) for(Rg int i=a;i>=b;i--)
using namespace std;
const int N=;
int ct[N],n;
ll f[N][N/+][N/+][N/+][N/+];
int main()
{
while(scanf("%d",&n),n)
{
memset(ct,,sizeof(ct));memset(f,,sizeof(f));
go(i,,n)scanf("%d",&ct[i]);
f[][][][][]=;
go(a,,ct[])
go(b,,ct[])
go(c,,ct[])
go(d,,ct[])
go(e,,ct[])
{
ll t=f[a][b][c][d][e];
if(a<ct[])f[a+][b][c][d][e]+=t;
if(b<ct[]&&a>b)f[a][b+][c][d][e]+=t;
if(c<ct[]&&b>c)f[a][b][c+][d][e]+=t;
if(d<ct[]&&c>d)f[a][b][c][d+][e]+=t;
if(e<ct[]&&d>e)f[a][b][c][d][e+]+=t;
}
printf("%llu\n",f[ct[]][ct[]][ct[]][ct[]][ct[]]);
}
return ;
}
随机推荐
- 快递查询API接口集成,有需要的可以直接用
适用于涉及经常发货.寄快递的人群.企业.电商网站.微信公众号平台等对接使用.支持国内外三百多家快递及物流公司的快递单号一站式查询. 使用说明: 1.KuadidiAPI.php 不需要修改改任何东西 ...
- Python深入:Distutils安装包
一:简介 尽管Python标准库已经支持很多的功能,但是有时还是会需要以第三方模块的形式添加新的功能到Python中. 在最简单的场景中,是你要安装的模块,已经有了针对你的平台的安装版本,因此只需要像 ...
- HZOJ 寿司
这题也是挺神仙的,现在O(n)的解法还没打出来,只是用O(nlogn)卡过去了(理论上可以过),sdfz某大佬用三分拿到了65分…… 考试连暴力都没打出来…… n2暴力T40: 首先将环拆成链,我们可 ...
- Python基础10 回顾
从最初的"Hello World",走到面向对象,该回过头来看看,教程中是否遗漏了什么. 我们之前提到一句话,"Everything is Object".那么 ...
- Python--day72--json内容回顾
前后端分离,Json格式字符串:序列化和反序列化 """ 复习python中的json模块 """ import json s = '{&q ...
- pytorch lstm crf 代码理解
好久没有写博客了,这一次就将最近看的pytorch 教程中的lstm+crf的一些心得与困惑记录下来. 原文 PyTorch Tutorials 参考了很多其他大神的博客,https://blog.c ...
- H3C 帧中继子接口
- java 利用反射创建对象
创建对象: 1.使用Class对象的newInstance()方法创建该Class对象的实例,此时该Class对象必须要有无参数的构造方法. 2.使用Class对象获取指定的Constructor对象 ...
- 11584 - Partitioning by Palindromes——[DP]
题意分析: 题目比较容易理解,以d[i]表示前i个字符的最优解,状态转移方程 d[i]=min{d[j]+1| [j+1~i]为回文串} 代码如下: #include <cstdio> # ...
- 2018-8-10-C#-局部函数与事件
title author date CreateTime categories C# 局部函数与事件 lindexi 2018-08-10 19:16:52 +0800 2018-2-13 17:23 ...