牛牛喜欢整数序列,他认为一个序列美丽的定义是
1:每个数都在0到40之间
2:每个数都小于等于之前的数的平均值
具体地说:for each i, 1 <= i < N, A[i] <= (A[0] + A[1] + … + A[i-1]) / i.
3:没有三个连续的递减的数
现在给你一个序列,每个元素是-1到40,你可以将序列中的-1修改成任意的数,求你可以得到多少个美丽序列,答案对1e9+7取模
输入描述:
第一行输入一个整数n (1 ≤ n ≤ 40)
第二行输入n个整数
输出描述:
输出一个整数
示例1
输入
2
3 -1
输出
4
示例2
输入
3
5 3 -1
输出
2
示例3
输入
3
-1 0 40
输出
0
示例4
输入
11
-1 40 -1 -1 -1 10 -1 -1 -1 21 -1
输出
579347890
看了一些题解,总结一下,采用动态规划,有几个条件就开几维数组
dp[i][j][flag][sum] 表示当第i个数时,如果这个数时j,l他和前面的数是否构成数组降序(1就是前面没有降序,2就是和前一个构成降序,题目中不超过三个降序),sum就是前面i-1个数的综合(所以这个数的循环从j*(i-1)开始循环,使得这个数要大于前面数的平均值)
分类讨论及其转移方程的思路分析:
(1)如果第一个数就为-1,我们可以从0循环到40,令dp[1][i][1][i]=1,因为-1可以变为任意一个数,所以第一项有41种可能的数字,总和也有41种可能,都为0到40。
//这里为对第一个数字是否为-1进行讨论
if(a[1]==-1)
{
for(ll i=0;i<=40;i++)
dp[1][i][1][i]=1;
}
else
dp[1][a[1]][1][a[1]]=1;
(2)如果第一项不是-1,就直接dp[1][a[1]][1][a[1]]=1即可。然后从数组的第二项开始循环
①如果当前数字为-1,当前数字可以为0到40一共41种可能,又因为当前数字的前一个数字我们不知道是多少,所以也可以理解为0到40一共41种可能,最后我们需要遍历总和,因为要保证每个数都小于等于之前的数的平均值,所以需要遍历j*(i-1)到1600-j,如果当前数字大于上一个数字,就有dp[i][j][1][sum+j]=(dp[i][j][1][sum+j]+dp[i-1][flag][1][sum]+dp[i-1][flag][2][sum])%mod,否则就有dp[i][j][2][sum+j]=(dp[i][j][2][sum+j]+dp[i-1][flag][1][sum])%mod
//这里为在循环中,如果当前数字为-1进行的操作
if(a[i]==-1)
{
for(ll j=0;j<=40;j++) //这里循环的是当前数字,-1可以随便变数,所以为0到40的循环
{
for(ll flag=0;flag<=40;flag++) //这里为当前数字的前一个数字进行0到40的循环
{
for(ll sum=j*(i-1);sum<=1600-j;sum++) //这里是对总和的遍历
{
if(j>=flag) //如果为当前数大于等于前一个数,则当前数为递减序列的第一个,上一个数可以为递减序列的第一个或是第二个。
dp[i][j][1][sum+j]=(dp[i][j][1][sum+j]+dp[i-1][flag][1][sum]+dp[i-1][flag][2][sum])%mod;
else //否则,当前数只能是递减序列的第二个,上一个数只能是递减序列的第一个
dp[i][j][2][sum+j]=(dp[i][j][2][sum+j]+dp[i-1][flag][1][sum])%mod;
}
}
}
}
②如果当前数字不为-1,我们可以根据为-1时的讨论,省去对当前数字的讨论,同时遍历总和时为a[i]*(i-1)到1600-a[i],这是如果当前数字大于上一个数字,就有dp[i][a[i]][1][sum+a[i]]=(dp[i][a[i]][1][sum+a[i]]+dp[i-1][flag][1][sum]+dp[i-1][flag][2][sum])%mod;否则就有dp[i][a[i]][2][sum+a[i]]=(dp[i][a[i]][2][sum+a[i]]+dp[i-1][flag][1][sum])%mod
//这里是当前数不为-1时的操作
else
{
for(ll flag=0;flag<=40;flag++) //这里是对上一个数字从0到40的遍历,省去了对当前数的讨论。
{
for(ll sum=a[i]*(i-1);sum<=1600-a[i];sum++) //这里是对总和的遍历
{
if(a[i]>=flag) //如果当前数大于上一个数,按照上一个代码中的思路进行转移即可。
{
dp[i][a[i]][1][sum+a[i]]=(dp[i][a[i]][1][sum+a[i]]+dp[i-1][flag][1][sum]+dp[i-1][flag][2][sum])%mod;
}
else //这里也按照上一个代码的思路进行转移
{
dp[i][a[i]][2][sum+a[i]]=(dp[i][a[i]][2][sum+a[i]]+dp[i-1][flag][1][sum])%mod;
}
}
}
}
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=1e6+7;
const int mod=1e9+7;
ll m,n,i,j,ans;
ll a[maxn];
ll dp[50][50][3][1605];
int main()
{
cin>>n;
for(ll i=1;i<=n;i++)
cin>>a[i];
//这里为对第一个数字是否为-1进行讨论
if(a[1]==-1)
{
for(ll i=0;i<=40;i++)
dp[1][i][1][i]=1;
}
else
dp[1][a[1]][1][a[1]]=1;
for(ll i=2;i<=n;i++)
{
//这里为在循环中,如果当前数字为-1进行的操作
if(a[i]==-1)
{
for(ll j=0;j<=40;j++) //这里循环的是当前数字,-1可以随便变数,所以为0到40的循环
{
for(ll flag=0;flag<=40;flag++) //这里为当前数字的前一个数字进行0到40的循环
{
for(ll sum=j*(i-1);sum<=1600-j;sum++) //这里是对总和的遍历
{
if(j>=flag) //如果为当前数大于等于前一个数,则当前数为递减序列的第一个,上一个数可以为递减序列的第一个或是第二个。
dp[i][j][1][sum+j]=(dp[i][j][1][sum+j]+dp[i-1][flag][1][sum]+dp[i-1][flag][2][sum])%mod;
else //否则,当前数只能是递减序列的第二个,上一个数只能是递减序列的第一个
dp[i][j][2][sum+j]=(dp[i][j][2][sum+j]+dp[i-1][flag][1][sum])%mod;
}
}
}
}
//这里是当前数不为-1时的操作
else
{
for(ll flag=0;flag<=40;flag++) //这里是对上一个数字从0到40的遍历,省去了对当前数的讨论。
{
for(ll sum=a[i]*(i-1);sum<=1600-a[i];sum++) //这里是对总和的遍历
{
if(a[i]>=flag) //如果当前数大于上一个数,按照上一个代码中的思路进行转移即可。
{
dp[i][a[i]][1][sum+a[i]]=(dp[i][a[i]][1][sum+a[i]]+dp[i-1][flag][1][sum]+dp[i-1][flag][2][sum])%mod;
}
else //这里也按照上一个代码的思路进行转移
{
dp[i][a[i]][2][sum+a[i]]=(dp[i][a[i]][2][sum+a[i]]+dp[i-1][flag][1][sum])%mod;
}
}
}
}
}
for(int j=0;j<=1600;j++)
{
for(int i=0;i<=40;i++)
{
ans=(ans+dp[n][i][1][j])%mod;
ans=(ans+dp[n][i][2][j])%mod;
}
}
cout<<ans<<endl;
return 0;
}
文章浏览阅读6.6k次,点赞15次,收藏13次。前言本系列包含两篇博客,主要分享OpenCV常用的操作,文末会分享相关著作电子版链接。传送门基础操作进阶操作资源链接:https://pan.baidu.com/s/1FlioORwwixsopfESgnsJpA提取码:aogg_opencv4快速入门pdf百度网盘
文章浏览阅读6.8k次。作为时序列数据库中表现很好的InfluxDB,已经被越来越多的项目应用到实际当中。比如资源的监控来说,Cadvisor监控到某一断面某一节点整体资源的状况,而且能够不断地从画面中看到不断变化的信息,但是这些数据如何持久化的保存是一个问题,而InfluxDB正好可以满足这个需求,其简单好用,耦合度小,容易集成到整体的系统之中。_数据库 断面查询
文章浏览阅读6.3k次。很久没有给大家带来家庭影院功放,特别是中低档的家庭影院功放试用了。要说我为什么要试用这台天龙AVR-X3600H,还真不是我主动去找的。这是影院君告诉我天龙最新的X3000系列功放也加入了7.2.4声道的前级输出能力,说这应该去找天龙借一台试一下。而我借这台功放的目的,是想证明天龙X6500H以下的型号,人声都偏薄!大家去买同一家公司的马兰士吧!高端和旗舰差多少?天龙AVR-X4500H与AVC-..._天龙avr功放等级排列
文章浏览阅读4.5k次,点赞2次,收藏12次。首先,来说一下什么是文件句柄。百度百科的解释是:在文件I/O中,要从一个文件读取数据,应用程序首先要调用操作系统函数并传送文件名,并选一个到该文件的路径来打开文件。该函数取回一个顺序号,即文件句柄(file handle),该文件句柄对于打开的文件是唯一的识别依据。要从文件中读取一块数据,应用程序需要调用函数ReadFile,并将文件句柄在内存中的地址和要拷贝的字节数传送给操作系统。当完成任务后,..._c# file句柄
文章浏览阅读3.4k次。最近在做项目的过程中遇到一个非常奇葩的bug,在h5页面点击一个按钮弹出弹窗,但是这个弹窗刚出现就会自动消失,导致屏幕出现闪动现象,关键这个bug还是偶现的。经过一番研究才发现是vue事件点击穿透引起的,而且弹窗一定要在300ms内出现才会引发这个bug,接下来分析具体原因:一,click与300ms延迟vue框架内置指令v-on:click有300ms的延迟响应,这是为了判断区分单击和双击。vu..._vue双事件
文章浏览阅读132次。dp优化_dp优化的学习路径
文章浏览阅读2.2k次,点赞33次,收藏48次。最全的acm算法竞赛赛制介绍、比赛、题目形式、常见测评状态_算法赛
文章浏览阅读5.9k次。最近有些朋友在看《深入浅出MFC》的时候,被第三章的几个宏给卡住了,记得我第一次看此书时,也被这几个宏给卡住。当然真正卡人的其实是第一个,也就是DECLARE_DYNAMIC/IMPLEMENT_DYNAMIC。我做了一个详解,供同样被卡住的朋友做个参考:)说明:这两个宏的主要目的,是在所指定的class(比如CView)的声明和实现里,加上一些静态成员函数和静态成员变量。所以,不要管“/”这个换_深入浅出 mfc declare 宏
文章浏览阅读3.2w次,点赞11次,收藏24次。图片转灰度cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)报错如下: img = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)cv2.error: OpenCV(4.5.4) d:\a\opencv-python\opencv-python\opencv\modules\imgproc\src\color.simd_helpers.hpp:92: error: (-2:Unspecified error) in function '__cdecl_cv2.cvtcolor报错
文章浏览阅读387次。一、 应用场合1、楼宇对讲–室内机2、智能家居–智能网关3、语音播放,视频播放机–带智能语音离线识别4、电梯广告机及楼层显示5、智能家居–86盒中控6、家电产品–破壁机,洗衣机等7、VOIP产品–SIP话机8、智能咖啡机9、打印机网关…二、 功能概述1、CPU:SSD201 ARM Cortex-A7 512Mbit DDR22、存储:SPI flash 128Mbit3、语音:支持一路数字音频输入和一路模拟音频输入4、音频:3W 音频放大输出5、串口:一路 485 输出,2 _ssd202 最高分辨率
文章浏览阅读781次。又是一年毕业季,每到_34岁测试女,该转型技术还是管理?
文章浏览阅读105次。临近节日,各个平台都在准备节目。最近就有网友发出一段视频,视频中有很多大家熟悉的实力派演员聚集在一起准备节目,认真彩排。比如陈宝国、冯远征、成龙、何冰,都是代表作响亮的实力派演员。从视频中可以看到,这些演员穿着统一的蓝色制服,虽然外形高矮不一,但是都是一身正气的帅气。值得一提的是在这一群老戏骨中,成龙站在了C位。从这个站位来看,基本也就反应了几人中的咖位。不过个子较小的成龙站在最中间莫名有一种喜感..._64位大年纪