洛谷 P3205 [HNOI2010] 合唱队

思路

先设 d p i j dpij dpij 为区间 i , j i, j i,j 的队形方案数。

考虑如何转移:对于区间 i , j i, j i,j 来说,最后一个入队的要么是 i i i,要么是 j j j。

所以分类讨论:

  1. 当 j j j 为最后一个入队的时, i i i 与 j − 1 j - 1 j−1 都可能是倒数第二个入队的。要满足的条件分别是 h i < h j hi < hj hi<hj 和 h j − 1 < h i hj - 1 < hi hj−1<hi
  2. 当 i i i 为最后一个入队的时, j j j 与 i + 1 i + 1 i+1 都可能是倒数第二个入队的。要满足的条件分别是 h i < h j hi < hj hi<hj 和 h i < h i + 1 hi < hi + 1 hi<hi+1

发现上一个状态有两种,即:在上一个状态中,最后一个进来的数(也就是当前状态的倒数第二个进来的数)是在【队尾/队头】。

因此我们要丰富一下状态定义:设 d p i j 0 dpij0 dpij0 表示最后一个进来的数在【队头】的方案数, d p i j 1 dpij1 dpij1 表示最后一个进来的数在【队尾】的方案数。最后答案就是 d p 1 n 0 + d p 1 n 1 dp1n0 + dp1n1 dp1n0+dp1n1

代码

cpp 复制代码
#include <bits/stdc++.h>

using namespace std;

const int maxn = 1e3 + 7;
const int mod  = 19650827;

int n, a[maxn];
int dp[maxn][maxn][2];
int main() {
	scanf("%d", &n);
	for (int i = 1; i <= n; ++i)
	    scanf("%d", a + i), 
		dp[i][i][0] = 1;

	for (int len = 2; len <= n; ++len) {
		for (int i = 1; i + len - 1 <= n; ++i) {
			int j = i + len - 1;
			
			if (a[i] < a[j]) dp[i][j][1] = (dp[i][j][1] + dp[i][j - 1][0]) % mod;
			if (a[j - 1] < a[j]) dp[i][j][1] = (dp[i][j][1] + dp[i][j - 1][1]) % mod;
		    
		    if (a[i] < a[j]) dp[i][j][0] = (dp[i][j][0] + dp[i + 1][j][1]) % mod;
			if (a[i] < a[i + 1]) dp[i][j][0] = (dp[i][j][0] + dp[i + 1][j][0]) % mod;
		}
	}
	printf("%d\n", (dp[1][n][0] + dp[1][n][1]) % mod);
	return 0;
} 
相关推荐
wabs6661 小时前
关于动态规划【0-1背包思想在实际问题中是怎么转化的?】
算法·动态规划
随意起个昵称2 小时前
区间dp-进阶题目1(进阶合并)
c++·算法·动态规划
2601_9618752415 小时前
法考资料2026|全套|资料已整理
数据结构·算法·链表·贪心算法·eclipse·线性回归·动态规划
汉克老师15 小时前
GESP2026年3月认证C++六级真题与解析(编程题1 选数)
c++·动态规划·线性dp·gesp六级·状态转移·选与不选
渡之17 小时前
GRiM-Net 深度解析 | 无人机 GNSS 拒止场景下两阶段跨视角视觉定位框架
深度学习·算法·动态规划·无人机
xhtdj19 小时前
Uber 如何通过批处理实现单账户每秒30+次更新
大数据·数据库·人工智能·安全·动态规划
wabs6661 天前
关于动态规划【纯粹的0-1背包需要思考的问题】
算法·动态规划
hai3152475431 天前
九章编程法 · 字典引擎【0/1拓扑步进 · 矩阵压缩·终极封版】
人工智能·数学建模·性能优化·动态规划·代码复审·傅立叶分析·极限编程
-森屿安年-1 天前
62. 不同路径
算法·动态规划
-森屿安年-2 天前
91. 解码方法
c++·动态规划