引入

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

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

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

所以我们就要用连续段 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. 红日安全vulnstack (一)

    网络拓扑图 靶机参考文章 CS/MSF派发shell 环境搭建 IP搭建教程 本机双网卡 65网段和83网段是自己本机电脑(虚拟机)中的网卡, 靶机外网的IP需要借助我们这两个网段之一出网 Kali ...

  2. BizWorks助力企业应用的高效开发与复用

    简介: BizWorks作为企业级云原生应用数字工作台,能很好地支撑企业数字中台建设.云原生应用开发.企业资产运营管理等场景.本文不会全面介绍BizWorks平台的能力,而是着重介绍BizWorks在 ...

  3. 基于信通院 Serverless 工具链模型的实践:Serverless Devs

    简介: Serverless Devs 作为开源开放的开发者工具,参编中国信通院<基于无服务器架构的工具链能力要求>标准,为行业统一规范发挥助推作用!​ 作者 | 江昱(阿里云 Serve ...

  4. Hologres揭秘:如何支持超高QPS在线服务(点查)场景

    简介: 本期我们将揭秘Hologres如何支持超高QPS在线服务(点查)场景. Hologres(中文名交互式分析)是阿里云自研的一站式实时数仓,这个云原生系统融合了实时服务和分析大数据的场景,全面兼 ...

  5. 庖丁解牛-图解MySQL 8.0优化器查询解析篇

    ​简介: SQL优化器本质上是一种高度抽象化的数据接口的实现,经过该设计,客户可以使用更通用且易于理解的SQL语言,对数据进行操作和处理,而不需要关注和抽象自己的数据接口,极大地解放了客户的应用程序. ...

  6. [FAQ] Fontconfig error: Cannot load default config file

      在使用一些第三方库时(比如生成图片),如果出现此提示,说明系统里缺少字体. 在 Ubuntu 上可以运行:$ apt-get install fontconfig 在 Centos 上可以运行:$ ...

  7. [FAQ] GoLand 需要手动开启代码补全吗 ?

    使用 go mod download 下载模块到本地缓存中,之后 GoLand 就会根据输入自动代码提示. Other:[FAQ] Goland 始终没有包代码的提示 Link:https://www ...

  8. dotnet 警惕 Assembly.Location 返回空

    在大部分情况下,获取当前所运行的应用程序的所在路径时,常用的就是 Assembly.Location 属性,按照之前的经验,使用 Assembly.Location 是最为稳定的做法,然而在 dotn ...

  9. dotnet 在 WPF 里显示数学 π 的颜色

    有逗比小伙伴问我,数学的 π 视觉效果是啥.于是我就来写一个逗比的应用将 π 的颜色在 WPF 应用画出来.原理就是读取 π 的小数点后的数值,然后使用逗比算法转换为 RGB 颜色像素,接着将这些像素 ...

  10. C++里也有菱形运算符?

    最近在翻<c++函数式编程>的时候看到有一小节在说c++14新增了"菱形运算符".我寻思c++里好像没什么运算符叫这名字啊,而且c++14新增的功能很少,我也不记得有添 ...