引入

在一类序列计数问题中,状态转移的过程可能与相邻的已插入元素的具体信息相关。

这类问题通常的特点是,如果只考虑在序列的一侧插入,问题将容易解决

枚举插入顺序的复杂度通常难以接受,转移时枚举插入位置又难以记录已插入元素的信息。

所以我们就要用连续段 dp。

dp 模型

连续段 dp 的好处在于,他的元素插入只会在连续段的两端进行。

所以他只会通过 建立新段,插入至已有连续段的两端,合并两段 来进行转移。

通常地,我们会按某种特定的顺序插入所有元素。

每次插入元素时,对三类转移方式进行分类讨论:

  1. 将插入的元素作为一个新连续段插入
  2. 将元素插入至一个已有连续段的两端
  3. 将元素用于合并两个连续段

分别会导致什么状态变化。

我们先从最基础的 dp 模型说起。

求满足某些限制的 \(n\) 个元素的排列数量

我们一般会定义 \(dp_{i,j}\) 为前 \(i\),形成了 \(j\) 个联通段的个数。

所以我们考虑三种情况。

  1. 建立新的连续段:\(dp_{i,j} \times (j + 1) \to dp_{i + 1,j + 1}\)
  2. 合并两个连续段:\(dp_{i,j} \times (j - 1) \to dp_{i + 1,j - 1}\)
  3. 插入至已有连续段的两端:\(dp_{i,j} \times 2 \times j \to dp_{i + 1,j}\)(注意此时有可能左右不同,所以要分讨)

可能有人会对转移方程有问题。我下面用一张直观一点的图做一个解释。

例题:

[COCI2021-2022#2] Magneti

考虑 DP。

令 \(g_k\) 为放完所有小球后还剩下 \(k\) 个空为的情况数。

所以容易推出 \(\sum^{m-n}_{k=0} g_k \times (^{n + m - k}_{\ \ \ \ \ \ k})\) 就是答案。

现在问题转化成了怎么求 \(g_k\)。

考虑 DP。

\(dp_{i,j,k}\) 表示放完前 \(i\) 个小球,有 \(j\) 个连续段,不能放球的位置有 \(k\) 个。

然后在上面的式子改一改即可。

  1. \((j + 1) \times dp_{i,j,k} \to dp_{i + 1,j + 1,k + 1}\)

  2. \((j - 1) \times dp_{i,j,k} \to dp_{i + 1,j - 1,k + 2\times r_{i+1} - 1}\)

  3. \(dp_{i,j,k} \times 2 \times j \to dp_{i + 1,j,k + r_{i + 1}}\)

所以就有代码啦!

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int mod = 1e9 + 7;
const int maxn = 55;
const int maxl = 10000 + 100;
int n,l;
int r[maxn],g[maxl];
int dp[maxn][maxn][maxl];
int fac[maxl],inv[maxl];
int qpow(int a,int b)
{
int res = 1;
while(b)
{
if(b & 1)
{
res = res * a % mod;
}
a = a * a % mod;
b >>= 1;
}
return res;
}
int C(int a,int b)
{
return fac[a] * inv[b] % mod * inv[a - b] % mod;
}
signed main()
{
fac[0] = inv[0] = 1;
for(int i = 1;i < maxl;i++)
{
fac[i] = fac[i - 1] * i % mod;
inv[i] = qpow(fac[i],mod - 2);
}
cin >> n >> l;
for(int i = 1;i <= n;i++)
{
cin >> r[i];
}
sort(r + 1,r + n + 1);
dp[0][0][0] = 1;
for(int i = 0;i < n;i++)
{
for(int j = 0;j <= i;j++)
{
for(int k = 0;k < l;k++)
{
if(k + r[i + 1] <= l)//插入在一个连续段的两端
{
dp[i + 1][j][k + r[i + 1]] = (dp[i + 1][j][k + r[i + 1]] + dp[i][j][k] * j * 2 % mod) % mod;
}
if(k + 2 * r[i + 1] - 1 <= l && j >= 2)//合并两个新段
{
dp[i + 1][j - 1][k + 2 * r[i + 1] - 1] = (dp[i + 1][j - 1][k + 2 * r[i + 1] - 1] + dp[i][j][k] * (j - 1) % mod) % mod;
}
if(k + 1 <= l)
{
dp[i + 1][j + 1][k + 1] = (dp[i + 1][j + 1][k + 1] + dp[i][j][k] * (j + 1) % mod) % mod;//增加一个段
}
}
}
}
for(int i = 0;i <= l;i++)
{
g[i] = dp[n][1][i];
}
int ans = 0;
for(int i = 0;i <= l;i++)
{
ans = (ans + C(l - i + n,n) * g[i] % mod) % mod;
}
cout << ans;
return 0;
}

连续段 dp - 状态转移时依赖相邻元素的序列计数问题的更多相关文章

  1. 社论 22.10.9 优化连续段dp

    CF840C 给定一个序列 \(a\),长度为 \(n\).试求有多少 \(1\) 到 \(n\) 的排列 \(p_i\),满足对于任意的 \(2\le i\le n\) 有 \(a_{p_{i-1} ...

  2. 【XSY3344】连续段 DP 牛顿迭代 NTT

    题目大意 对于一个长度为 \(n\) 的排列 \(p\),我们称一个区间 \([l,r]\) 是连续的当且仅当 \((\max_{l\leq i\leq r}a_i)-(\min_{l\leq i\l ...

  3. hdu-4507 吉哥系列故事——恨7不成妻 数位DP 状态转移分析/极限取模

    http://acm.hdu.edu.cn/showproblem.php?pid=4507 求[L,R]中不满足任意条件的数的平方和mod 1e9+7. 条件: 1.整数中某一位是7:2.整数的每一 ...

  4. Bomb HDU 3555 dp状态转移

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=3555 题意: 给出一个正整数N,求出1~N中含有数字“49”的数的个数 思路: 采用数位dp的状态转移方程 ...

  5. HDU - 1176 免费馅饼 DP多种状态转移

    免费馅饼 都说天上不会掉馅饼,但有一天gameboy正走在回家的小径上,忽然天上掉下大把大把的馅饼.说来gameboy的人品实在是太好了,这馅饼别处都不掉,就掉落在他身旁的10米范围内.馅饼如果掉在了 ...

  6. 背包DP 存在异或条件的状态转移问题

    题目链接 分析:有大佬说可以用线性基写,可惜我不会,这是用DP写的 题目明确说明可到达的位置只与能值有关,和下标无关,我们就可以排个序,这样每个数可以转移的区间就是它的所有后缀 我们可以用dp[i][ ...

  7. 状压dp终极篇(状态转移的思想)

    状压dp是将每种状态都压缩成用一个二进制串,然后利用位运算进行操作的dp,而凡是dp都需要进行状态转移 对于简单的dp问题只需要一个二维数组dp[ i ][ j ]就能解决 具体操作为首先把状态压缩为 ...

  8. [总结-动态规划]经典DP状态设定和转移方程

    马上区域赛,发现DP太弱,赶紧复习补上. #普通DP CodeForces-546D Soldier and Number Game 筛法+动态规划 待补 UVALive-8078 Bracket S ...

  9. (leetcode:选择不相邻元素,求和最大问题):打家劫舍(DP:198/213/337)

    题型:从数组中选择不相邻元素,求和最大 (1)对于数组中的每个元素,都存在两种可能性:(1)选择(2)不选择,所以对于这类问题,暴力方法(递归思路)的时间复杂度为:O(2^n): (2)递归思路中往往 ...

  10. dp状态压缩

    dp状态压缩 动态规划本来就很抽象,状态的设定和状态的转移都不好把握,而状态压缩的动态规划解决的就是那种状态很多,不容易用一般的方法表示的动态规划问题,这个就更加的难于把握了.难点在于以下几个方面:状 ...

随机推荐

  1. 力扣49(java)-字母异位词分组(中等)

    题目: 给你一个字符串数组,请你将 字母异位词 组合在一起.可以按任意顺序返回结果列表. 字母异位词 是由重新排列源单词的字母得到的一个新单词,所有源单词中的字母通常恰好只用一次. 示例 1: 输入: ...

  2. 宏杉科技加入阿里云PolarDB开源数据库社区

    简介: 宏杉科技签署阿里巴巴开源CLA(Contribution License Agreement, 贡献许可协议), 正式与阿里云PolarDB 开源数据库社区牵手. 宏杉科技签署阿里巴巴开源CL ...

  3. 快速界定故障:Socket Tracer网络监控实践

    ​ 简介: Socket Tracer定位是传输层(Socket&TCP)的指标采集工具,通过补齐网络监控的这部分盲区,来达到快速界定网络问题的目标. ​ 作者 | 四忌 来源 | 阿里技术公 ...

  4. dotnet UNO 如何在调试下输出界面层级结构

    本文将告诉大家如何在 UNO 里面将界面的层级结构输出到调试窗口 实现方法非常简单,和 WPF 或 UWP 等的方法是一样的,那就是通过可视化树遍历的方式,如以下代码 static class UIS ...

  5. dotnet 通过 DockerfileContext 解决项目放在里层文件夹导致 VisualStudio 构建失败

    本文告诉大家,如何解决 csproj 项目文件放入到里层的文件夹,不放在 sln 所在文件夹的第一层子文件夹,导致 VisualStudio 2022 在构建 docker 映像提示找不到文件的问题 ...

  6. MSBuild 输出日志可视化工具 MSBuild Structured Log Viewer 简介

    感谢 Vatsan Madhavan 小伙伴推荐的 MSBuild 输出日志可视化工具,这个工具可以使用漂亮的 WPF 界面预览 MSBuild 复杂的输出内容 这是一个完全开源的工具,请看 Kiri ...

  7. 2019-8-31-NuGet-如何设置图标

    title author date CreateTime categories NuGet 如何设置图标 lindexi 2019-08-31 16:55:58 +0800 2019-4-27 17: ...

  8. 二:大数据架构回顾-Kappa架构

    Kappa 架构是由 LinkedIn 的前首席工程师杰伊·克雷普斯(Jay Kreps)提出的一种架构思想.克雷普斯是几个著名开源项目(包括 Apache Kafka 和 Apache Samza ...

  9. postgresql用sql查询表结构

    查询sql如下: SELECT a.attname AS field, t.typname AS type, CASE WHEN t.typlen = -1 THEN a.atttypmod - 4 ...

  10. 海康iSC综合安防平台-视频web插件调试

    综合安防管理平台 视频WEB插件 1.demo_window_simple_playback.html.demo_window_simple_preview.html为简化版demo,可在此基础上开发 ...