美丽序列(Dp)

这篇具有很好参考价值的文章主要介绍了美丽序列(Dp)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

传送门

题目描述

牛牛喜欢整数序列,他认为一个序列美丽的定义是
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

4

示例2

输入

复制

3
5 3 -1

输出

复制

2

示例3

输入

复制

3
-1 0 40

输出

复制0

0

示例4

输入

复制

11
-1 40 -1 -1 -1 10 -1 -1 -1 21 -1

输出

复制

579347890

备注:

子任务1: n <= 10
子任务2: n <= 20
子任务3: 无限制

思路:

首先状态设计:dp[i][j][k][s]

代表当前是第i个数字,且第i位选择数字j,

选择数字j后形成的递减序列长度为k

前i个数字的和为s

此时的方案数为dp[i][j][k][s]

显然分析,k只有1和2两种情况

1的时候就是当前选的数字比上一个大(所以说是当前递减数列的第一个数字)

2的时候就是当前选的数字比上一个小(不可能出现比上两个小的情况,因为这样序列长度就是3不合题意了)

那么状态转移方程也能写了

我们需要枚举一下上一个数字的值

依靠这个进行转移

先不管这个数字是否可以自定(如果要自定就再加一层循环枚举这个数字)

如果上一个的值大于当前值

dp[i][当前值][2][之前的和+当前值]+=dp[i-1][上一位值][1][之前的和]

因为这个时候形成递减数列了,所以递减数列的长度是2,就只转移到长度为2的状态

如果当前值大于等于上一个值

dp[i][当前值][1][之前的和+当前值]+=dp[i-1][之前值][1][之前和]+dp[i-1][之前值][2][之前和]

讨论一下枚举的界限问题

AC代码:文章来源地址https://www.toymoban.com/news/detail-537196.html

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N	=50;
const int mod=1e9+7;
int a[N];
int dp[45][45][3][1605];
void solve(){
	int n;
	cin>>n;
	for(int i=1;i<=n;i++){
		cin>>a[i];
	}
	if(a[1]==-1){
		for(int i=0;i<=40;i++){
			dp[1][i][1][i]=1;
		}
	}
	else dp[1][a[1]][1][a[1]]=1;
	for(int i=2;i<=n;i++){
		if(a[i]==-1){
			for(int j=0;j<=40;j++){
				for(int k=0;k<=40;k++){
					for(int s=j*(i-1);s<=1600-j;s++){
						if(j>=k){ 
							dp[i][j][1][s+j]=(dp[i][j][1][s+j]+dp[i-1][k][1][s])%mod;
							dp[i][j][1][s+j]=(dp[i][j][1][s+j]+dp[i-1][k][2][s])%mod;
						}
						else {
							dp[i][j][2][s+j]=(dp[i][j][2][s+j]+dp[i-1][k][1][s])%mod;	
						} 
					}
				}
			} 
		}
		else {
			for(int k=0;k<=40;k++){
				for(int s=a[i]*(i-1);s<=1600-a[i];s++){
					if(a[i]>=k){
						dp[i][a[i]][1][s+a[i]]=(dp[i][a[i]][1][s+a[i]]+dp[i-1][k][1][s])%mod;
						dp[i][a[i]][1][s+a[i]]=(dp[i][a[i]][1][s+a[i]]+dp[i-1][k][2][s])%mod;
					}
					else{
						dp[i][a[i]][2][s+a[i]]=(dp[i][a[i]][2][s+a[i]]+dp[i-1][k][1][s])%mod;
					}
				}
			}
		}
	}
	int sum=0;
	for(int i=0;i<=40;i++){
		for(int j=n*i;j<=1600;j++){
			sum=(sum+dp[n][i][1][j])%mod;
			sum=(sum+dp[n][i][2][j])%mod;
		}
	}
	cout<<sum<<'\n';
}
signed main(){
	ios::sync_with_stdio(false),cin.tie(0),cout.tie(0);
	int t=1;
//	cin>>t;
	while(t--){
		solve();
	}
}

到了这里,关于美丽序列(Dp)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【数据结构】八大排序算法(内含思维导图和画图分析)

    作者主页: paper jie_博客 本文作者:大家好,我是paper jie,感谢你阅读本文,欢迎一建三连哦。 本文录入于《JAVA数据结构》专栏,本专栏是针对于大学生,编程小白精心打造的。笔者用重金(时间和精力)打造,将javaSE基础知识一网打尽,希望可以帮到读者们哦。 其他专栏:

    2024年02月08日
    浏览(63)
  • 周赛348(模拟、反向思维、数位DP)

    难度中等3 给你一个下标从 0 开始的字符串 s ,重复执行下述操作 任意 次: 在字符串中选出一个下标 i ,并使 c 为字符串下标 i 处的字符。并在 i 左侧 (如果有)和 右侧 (如果有)各 删除 一个距离 i 最近 的字符 c 。 请你通过执行上述操作任意次,使 s 的长度 最小化 。

    2024年02月07日
    浏览(74)
  • 数据结构与算法 -- 子序列问题

    一、子序列问题         子数组问题是连续的,而子序列问题是不连续的。比如说字符串 “I wanna keep a giraffe in my backyard” 的一种子序列就可以是 “Igbackd”。 子序列不再要求答案是一个连续的串。一个字符串的子序列,是由原字符串在不改变字符的相对顺序的情况下删

    2024年02月09日
    浏览(54)
  • python数据结构与算法-动态规划(最长公共子序列)

    一个序列的子序列是在该序列中删去若干元素后得 到的序列。 例如:\\\"ABCD”和“BDF”都是“ABCDEFG”的子序列。 最长公共子序列(LCS) 问题: 给定两个序列X和Y,求X和Y长度最大的公共子字列。 例:X=\\\"ABBCBDE”Y=\\\"DBBCDB”LCS(XY)=\\\"BBCD\\\" 应用场景:字符串相似度比对 (1)问题思考 思考: 暴

    2024年02月08日
    浏览(53)
  • 算法与数据结构(二十三)动态规划设计:最长递增子序列

    注:此文只在个人总结 labuladong 动态规划框架,仅限于学习交流,版权归原作者所有; 也许有读者看了前文 动态规划详解,学会了动态规划的套路:找到了问题的「状态」,明确了 dp 数组/函数的含义,定义了 base case;但是不知道如何确定「选择」,也就是找不到状态转移

    2024年02月13日
    浏览(43)
  • 动态规划dp —— 28.摆动序列

    连续相同的数不算是摆动序列 单独一个或不相等的两个数算是摆动序列  是什么?dp表中里的值所表示的含义就是状态表示 dp[i]表示:以i位置为结尾的所有子序列中,最长的摆动序列的长度 但是i位置的值可能是下降后的,也可能是上升后的,所以细分为两种状态表示 f[i]表

    2024年02月12日
    浏览(36)
  • 【重点】【DP】300. 最长递增子序列

    题目 更好的方法是耐心排序,参见《算法小抄》的内容!!! 基础解法必须掌握!!!

    2024年01月17日
    浏览(57)
  • 数据结构与算法——希尔排序(引例、希尔增量序列、原始希尔排序、代码、时间复杂度、Hibbard增量序列、Sedgewick增量序列)

    目录 引例 希尔增量序列 原始希尔排序 代码(C语言) 时间复杂度 更多增量序列 Hibbard增量序列 Sedgewick增量序列 希尔排序(by Donald Shell) 给以下序列进行排序:  先以5的间隔进行插入排序:  再以3的间隔进行插入排序: 最后再以1为间隔做插入排序,即常规插入排序 ,得

    2024年02月16日
    浏览(50)
  • Codeforces Round 768 (Div. 1) D. Flipping Range(思维题 等价类性质 dp)

    题目 思路来源 官方题解 洛谷题解 题解 可操作的最短区间长度肯定是gcd,记为g,然后考虑如何dp 考虑g个等价类,每个等价类i,i+g,i+2*g,... 每次翻转长度为g的区间,会同时影响到g个等价类总的翻转的奇偶性, 性质一:只有每个等价类翻的次数奇偶性相同才合法  性质二:此

    2024年01月19日
    浏览(35)
  • 【算法与数据结构】106、LeetCode从中序与后序遍历序列构造二叉树

    所有的LeetCode题解索引,可以看这篇文章——【算法和数据结构】LeetCode题解。    思路分析 :首先我们要知道后序遍历数组的最后一个元素必然是根节点,然后根据根节点在中序遍历数组中的位置进行划分,得到根节点的左右子树遍历数组,以此递归。当然这里有一个前提

    2024年02月10日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包