牛客_21313美丽序列_动态规划_一行两个整数,表示最美丽的段的起始下标和终止下标,如果有多段最美丽的段,请输出-程序员宅基地

技术标签: 算法  动态规划  C/C++题目记录  

21313美丽序列

题目描述

牛牛喜欢整数序列,他认为一个序列美丽的定义是
1:每个数都在040之间
2:每个数都小于等于之前的数的平均值
具体地说:for each i, 1 <= i < N, A[i] <= (A[0] + A[1] ++ A[i-1]) / i.
3:没有三个连续的递减的数

现在给你一个序列,每个元素是-140,你可以将序列中的-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;
}

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_45619006/article/details/114684101

智能推荐

一文搞定OpenCV快速入门(附网盘链接)_opencv4快速入门pdf百度网盘-程序员宅基地

文章浏览阅读6.6k次,点赞15次,收藏13次。前言本系列包含两篇博客,主要分享OpenCV常用的操作,文末会分享相关著作电子版链接。传送门基础操作进阶操作资源链接:https://pan.baidu.com/s/1FlioORwwixsopfESgnsJpA提取码:aogg_opencv4快速入门pdf百度网盘

时序列数据库之InfluxDB_数据库 断面查询-程序员宅基地

文章浏览阅读6.8k次。作为时序列数据库中表现很好的InfluxDB,已经被越来越多的项目应用到实际当中。比如资源的监控来说,Cadvisor监控到某一断面某一节点整体资源的状况,而且能够不断地从画面中看到不断变化的信息,但是这些数据如何持久化的保存是一个问题,而InfluxDB正好可以满足这个需求,其简单好用,耦合度小,容易集成到整体的系统之中。_数据库 断面查询

功放前级的左右_相当超值的中档天龙家庭影院功放!天龙AVR-X3600H功放试用-程序员宅基地

文章浏览阅读6.3k次。很久没有给大家带来家庭影院功放,特别是中低档的家庭影院功放试用了。要说我为什么要试用这台天龙AVR-X3600H,还真不是我主动去找的。这是影院君告诉我天龙最新的X3000系列功放也加入了7.2.4声道的前级输出能力,说这应该去找天龙借一台试一下。而我借这台功放的目的,是想证明天龙X6500H以下的型号,人声都偏薄!大家去买同一家公司的马兰士吧!高端和旗舰差多少?天龙AVR-X4500H与AVC-..._天龙avr功放等级排列

如何通过C#来操作文件句柄_c# file句柄-程序员宅基地

文章浏览阅读4.5k次,点赞2次,收藏12次。首先,来说一下什么是文件句柄。百度百科的解释是:在文件I/O中,要从一个文件读取数据,应用程序首先要调用操作系统函数并传送文件名,并选一个到该文件的路径来打开文件。该函数取回一个顺序号,即文件句柄(file handle),该文件句柄对于打开的文件是唯一的识别依据。要从文件中读取一块数据,应用程序需要调用函数ReadFile,并将文件句柄在内存中的地址和要拷贝的字节数传送给操作系统。当完成任务后,..._c# file句柄

vue 点击一个按钮触发两个事件_vue事件点击穿透解决大法-程序员宅基地

文章浏览阅读3.4k次。最近在做项目的过程中遇到一个非常奇葩的bug,在h5页面点击一个按钮弹出弹窗,但是这个弹窗刚出现就会自动消失,导致屏幕出现闪动现象,关键这个bug还是偶现的。经过一番研究才发现是vue事件点击穿透引起的,而且弹窗一定要在300ms内出现才会引发这个bug,接下来分析具体原因:一,click与300ms延迟vue框架内置指令v-on:click有300ms的延迟响应,这是为了判断区分单击和双击。vu..._vue双事件

dp优化入门学习_dp优化的学习路径-程序员宅基地

文章浏览阅读132次。dp优化_dp优化的学习路径

随便推点

ACM算法竞赛入门——算法竞赛赛制、题目形式、常见评测状态_算法赛-程序员宅基地

文章浏览阅读2.2k次,点赞33次,收藏48次。最全的acm算法竞赛赛制介绍、比赛、题目形式、常见测评状态_算法赛

《深入浅出MFC》书中DECLARE_DYNAMIC/IMPLEMENT_DYNAMIC宏的详细解释_深入浅出 mfc declare 宏-程序员宅基地

文章浏览阅读5.9k次。最近有些朋友在看《深入浅出MFC》的时候,被第三章的几个宏给卡住了,记得我第一次看此书时,也被这几个宏给卡住。当然真正卡人的其实是第一个,也就是DECLARE_DYNAMIC/IMPLEMENT_DYNAMIC。我做了一个详解,供同样被卡住的朋友做个参考:)说明:这两个宏的主要目的,是在所指定的class(比如CView)的声明和实现里,加上一些静态成员函数和静态成员变量。所以,不要管“/”这个换_深入浅出 mfc declare 宏

图片转灰度报错cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)_cv2.cvtcolor报错-程序员宅基地

文章浏览阅读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报错

SigmaStar SSD201 SSD202室内网关开发板防雷防静电图_ssd202 最高分辨率-程序员宅基地

文章浏览阅读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 最高分辨率

技术还是管理,年过30的程序员何去何从?_34岁测试女,该转型技术还是管理?-程序员宅基地

文章浏览阅读781次。又是一年毕业季,每到_34岁测试女,该转型技术还是管理?

64位 c++分配 最大内存_成龙冯远征陈宝国同框,成龙稳居C位,64岁陈宝国霸气十足...-程序员宅基地

文章浏览阅读105次。临近节日,各个平台都在准备节目。最近就有网友发出一段视频,视频中有很多大家熟悉的实力派演员聚集在一起准备节目,认真彩排。比如陈宝国、冯远征、成龙、何冰,都是代表作响亮的实力派演员。从视频中可以看到,这些演员穿着统一的蓝色制服,虽然外形高矮不一,但是都是一身正气的帅气。值得一提的是在这一群老戏骨中,成龙站在了C位。从这个站位来看,基本也就反应了几人中的咖位。不过个子较小的成龙站在最中间莫名有一种喜感..._64位大年纪

推荐文章

热门文章

相关标签