自从多校后心憔悴啊,发现DP还是太水了,有一场的区间DP竟然不会做,咳,果然是赤裸裸的水军。

花了几天时间写了几道区间DP的题目,大部分都是水题,然后和以前的合并起来就是KB区间DP这个8 + 1道题的专辑,大家可以试着AK。

区间DP是一类在区间上进行动态规划的最优问题,一般是根据问题设出一个表示状态的dp,可以是二维的也可以是三维的,一般情况下为二维。然后将问题划分成两个子问题,也就是一段区间分成左右两个区间,然后将左右两个区间合并到整个区间,或者说局部最优解合并为全局最优解,然后得解。

这类DP可以用常规的for循环来写,也可以用记忆化搜索来写,个人更倾向于记忆化搜索写法,因为这种写法相当易懂,要什么值你直接去记忆化搜索一下就ok了,随叫随到随用啊。

1、ZOJ 3537 Cake

第一次写的 凸包结合 DP,调了好久,

问题属于经典的三角剖分

区间DP状态转移方程:

\[f[i][j] = min(f[i][k] + f[k][j] + cost[i][k] + cost[k][j]
\]

其中 \(j >= i+ 3,i+1<=k<=j-1,cost[i][k]\) 为连一条 \(i\) 到 \(k\) 的线的费用

详细题解:Here

2、LightOJ 1422 Halloween Costumes

本题不可以直接模拟栈是因为,不知道当前的衣服本来就有还是需要新穿一件。

初始化DP为一直穿衣服

for (int i = 1; i <= n; ++i) for (int j = i; j <= n; ++j)
f[i][j] = f[i][j - 1] + (j == i || a[j] != a[j - 1]);

然后从按区间DP板子,枚举 \(k\) ,只要 \(a[k]\) 与 \(a[j]\) 相同说明,是可以把从 \(k+1\) 到 \(j-1\) 的所有衣服都脱掉,然后 \(k\) 天的这件衣服,直接在第 \(j\) 天参加聚会,就不需要穿新的衣服。从小区间依次递推到大区间,就可以得到正确的答案。

\[f[i][j] = f[i][j - 1] (a[i] == a[j]) \\
f[i][j] = min(f[i][k] + f[k + 1][j]) (a[i] == a[k]\ and\ i\le k\le j)
\]
const int N = 110;
int a[N], n, Case, f[N][N];
int main() {
cin.tie(nullptr)->sync_with_stdio(false);
int _; for (cin >> _; _--;) {
cin >> n;
for (int i = 1; i <= n; ++i) cin >> a[i];
for (int i = 1; i <= n; ++i)
for (int j = i; j <= n; ++j) f[i][j] = f[i][j - 1] + (j == i || a[j] != a[j - 1]);
for (int len = 1; len <= n; ++len)
for (int i = 1; i + len <= n; ++i) {
int j = i + len;
for (int k = i; k + 1 <= j; ++k)
if (a[k] == a[j]) f[i][j] = min(f[i][j], f[i][k] + f[k + 1][j - 1]);
else f[i][j] = min(f[i][j], f[i][k] + f[k + 1][j - 1] + 1);
}
cout << "Case " << ++Case << ": " << f[1][n] << "\n";
memset(f, 0, sizeof(f));
}
}

3、POJ 2955 Brackets

先初始化一下匹配括号,一对匹配的括号肯定是他内部的答案加上2,然后再考虑一坨括号能不能由两坨括号拼接而成,如果可以,那么取他们的最大值。简而言之,本题就是两个条件,逐步去实现他们即可。

const int N = 110;
int f[N][N];
string s;
bool match(char a, char b) {
if ((a == '(' && b == ')') || (a == '[' && b == ']')) return true;
return false;
}
int main() {
while (cin >> s) {
if (s == "end") break;
memset(f, 0, sizeof(f));
int n = int(s.size());
for (int len = 1; len < s.size(); ++len)
for (int i = 0; i + len < s.size(); ++i) {
int j = len + i;
if (match(s[i], s[j]))
f[i][j] = f[i + 1][j - 1] + 2;
for (int p = i; p <= j; ++p)
f[i][j] = max(f[i][j], f[i][p] + f[p + 1][j]);
}
cout << f[0][s.size() - 1] << "\n";
}
}

4、CF 149 D Coloring Brackets

这题是上一个题目的升级版。

题面有三个要求:

1.匹配的括号必须涂其中一个并且不能都涂

2.相邻两个括号不能都染色,但是都可以不染色

const int mod = 1e9 + 7, N = 800;
string s;
stack<int>st;
ll m[N], f[N][N][3][3], n;
ll dfs(int a, int b, int c, int d) {
//a表示左端点位置 b表示右端点位置 c表示左端点颜色 d表示右端点颜色
//0表示没涂色 1表示涂了一种颜色 2表示涂了另一种颜色
ll res = 0;
if (b <= a || f[a][b][c][d] != -1) { //如果区间已经被分的不能再分,或者上次已经算过,返回
if (b <= a)//只有一种方案
return 1;
else
return f[a][b][c][d];
}
if (b == m[a]) { //遇到这个区间左右端点恰好匹配 那么左右端点的方案已经固定了 向内部推一层
if (c != 1) res += dfs(a + 1, b - 1, 1, 0) % mod;
//如果左端点颜色不为1,则内部相邻的左端点颜色可以为1,左端点涂了色,右端点一定不能涂色
if (d != 1) res += dfs(a + 1, b - 1, 0, 1) % mod;
//如果右端点颜色不为1,则同理
if (c != 2) res += dfs(a + 1, b - 1, 2, 0) % mod;
//如果左端点颜色不为2,则同理
if (d != 2) res += dfs(a + 1, b - 1, 0, 2) % mod;
//如果右端点颜色不为2,则同理
} else {
//这个区间左右端点不匹配,则有四种情况
res += (dfs(a + 1, m[a] - 1, 0, 1) * dfs(m[a] + 1, b, 1, d)) % mod;
res += (dfs(a + 1, m[a] - 1, 0, 2) * dfs(m[a] + 1, b, 2, d)) % mod;
ll p = dfs(m[a] + 1, b, 0, d);
if (c != 1) res += (p * dfs(a + 1, m[a] - 1, 1, 0)) % mod;
if (c != 2) res += (p * dfs(a + 1, m[a] - 1, 2, 0)) % mod;
}
f[a][b][c][d] = res % mod;//记录已经算好的
return res % mod;
}
int main() {
// cin.tie(nullptr)->sync_with_stdio(false);
while (getline(cin, s)) {
n = s.size();
for (int i = 0; i < n; ++i) {
if (s[i] == '(') st.push(i);
else {
m[st.top()] = i;
st.pop();
}
}
memset(f, -1, sizeof(f)); // 进行记忆化搜索之前,将dp数组置为-1,避免有0的方案出现却未被计算
cout << (dfs(0, n - 1, 3, 3)) << "\n";
}
}

5、1651 Multiplication Puzzle

由于题目说了,最左最右是不可以取走的,那么我们只用对第2到n-1这个区间的数进行讨论即可。

对于样例模拟一下拿数,拿数有以下顺序:

2 3 4

2 4 3

3 2 4

3 4 2

4 2 3

4 3 2

可见,拿数的方式并不一定是连续的。

所以,先初始化一下长度为1的区间,答案就是他本身乘以他旁边的两个数,合并区间的时候可能是两个连续的区间合并,也有可能是两个断开的区间再把断点对应的得分带上(断点位置最后拿),即可满足无后效性

const int N = 110;
ll f[N][N], a[N], n;
int main() {
cin >> n;
for (int i = 1; i <= n; ++i) cin >> a[i];
for (int i = 2; i <= n - 1; ++i) f[i][i] = a[i] * a[i - 1] * a[i + 1];
for (int len = 1; len <= n; ++len)
for (int i = 2; i + len <= n; ++i) {
int j = len + i;
f[i][j] = min(f[i + 1][j] + a[i] * a[i - 1] * a[j + 1], f[i][j - 1] + a[j] * a[i - 1] * a[j + 1]);
for (int k = i + 1; k <= j - 1; ++k)
f[i][j] = min(f[i][j], f[i][k - 1] + f[k + 1][j] + a[k] * a[i - 1] * a[j + 1]);
}
cout << f[2][n - 1] << "\n";
}

6、ZOJ 3469 Food Delivery

可以把所有点分为两组,一组是在起点左边的,一组是在起点右边的,用f[i][j][k]表示左边的送完了i个,右边的送完了j个,当前停留在k组(0表示左边那组,1表示右边那组),然后用记忆化搜索来进行递推

const int N = 1e3 + 10, inf = 0x3f3f3f3f;
struct node {int x, b;} pz[N], py[N]; //左边的点靠右排前
bool cmp(node a, node b) {return a.x < b.x;}
//左边的点走了第几个 右边的点走了第几个 如今停留在右边还是左边
ll f[N][N][2];
ll dp(int i, int j, int k) {
if (f[i][j][k] != -1) return f[i][j][k];
ll ans = inf;
if (k == 0) { // 往左边
if (i != 0) {
int tmp = pz[i].b + py[j + 1].b;//所有还未走的的人的愤怒值之和
//cout << tmp << endl;
ans = min(ans, dp(i - 1, j, 0) + (pz[i].x - pz[i - 1].x) * tmp);
ans = min(ans, dp(i - 1, j, 1) + (pz[i].x + py[j].x) * tmp);
}
} else {
if (j != 0) {
int tmp = pz[i + 1].b + py[j].b;
ans = min(ans, dp(i, j - 1, 1) + (py[j].x - py[j - 1].x) * tmp);
ans = min(ans, dp(i, j - 1, 0) + (pz[i].x + py[j].x) * tmp);
}
}
return f[i][j][k] = ans;
}
int main() {
cin.tie(nullptr)->sync_with_stdio(false);
int n, v, x;
while (cin >> n >> v >> x) {
int idz = 0, idy = 0;
for (int i = 1; i <= n; ++i) {
int tx, tb;
cin >> tx >> tb;
//直接存储相对位置
if (tx < x) pz[++idz].x = x - tx, pz[idz].b = tb;
if (tx > x) py[++idy].x = tx - x, py[idy].b = tb;
}
sort(pz + 1, pz + 1 + idz, cmp);
sort(py + 1, py + 1 + idy, cmp);
py[idy + 1].b = 0;
pz[idz + 1].b = 0;
for (int i = idy; i >= 1; --i) py[i].b += py[i + 1].b;
for (int i = idz; i >= 1; --i) pz[i].b += pz[i + 1].b;
memset(f, -1, sizeof(f));
f[0][0][0] = f[0][0][1] = 0;
cout << (v * min(dp(idz, idy, 0), dp(idz, idy, 1))) << "\n";
}
}

7、HDU 4283 You Are the One (较难)

8、Sdut 不老的传说问题 (较难)

9、HDU String painter

KB专题:区间DP专辑的更多相关文章

  1. kuangbin专题十二 POJ3186 Treats for the Cows (区间dp)

    Treats for the Cows Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7949   Accepted: 42 ...

  2. 「kuangbin带你飞」专题二十二 区间DP

    layout: post title: 「kuangbin带你飞」专题二十二 区间DP author: "luowentaoaa" catalog: true tags: - ku ...

  3. 区间dp专题练习

    区间dp专题练习 题意 1.Equal Sum Partitions ? 这嘛东西,\(n^2\)自己写去 \[\ \] \[\ \] 2.You Are the One 感觉自己智力被吊打 \(dp ...

  4. [kuangbin带你飞]专题二十二 区间DP

            ID Origin Title   17 / 60 Problem A ZOJ 3537 Cake   54 / 105 Problem B LightOJ 1422 Hallowee ...

  5. 专题训练之区间DP

    例题:以下例题部分的内容来自https://blog.csdn.net/my_sunshine26/article/details/77141398 一.石子合并问题 1.(NYOJ737)http: ...

  6. UESTC 2015dp专题 A 男神的礼物 区间dp

    男神的礼物 Time Limit: 20 Sec  Memory Limit: 256 MB 题目连接 http://acm.uestc.edu.cn/#/contest/show/65 Descri ...

  7. 【专题】区间dp

    1.[nyoj737]石子合并 传送门:点击打开链接 描述    有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这 ...

  8. 区间dp专题

    HDU4283You Are the One区间dp, 记忆话搜索运行时间:   #include <iostream> #include <cstdio> #include ...

  9. hdu4597 区间dp

    //Accepted 1784 KB 78 ms //区间dp //dp[l1][r1][l2][r2] 表示a数列从l1到r1,b数列从l2到r2能得到的最大分值 // #include <c ...

  10. hdu4283 区间dp

    //Accepted 300 KB 0 ms //区间dp //dp[i][j] 表示i到j第一个出场的最小diaosizhi //对于i到j考虑元素i //(1)i第一个出场,diaosizhi为 ...

随机推荐

  1. 🔥🔥Java开发者的Python快速进修指南:面向对象基础

    当我深入学习了面向对象编程之后,我首先感受到的是代码编写的自由度大幅提升.不同于Java中严格的结构和约束,Python在面向对象的实现中展现出更加灵活和自由的特性.它使用了一些独特的关键字,如sel ...

  2. js 通过id、pid遍历集合获得树结构

    原数据 let adreeJson = [ {id: 1, name: '陕西省', pid: 0}, {id: 2, name: '山西省', pid: 0}, {id: 3, name: '广东省 ...

  3. Silverlight工作流控件功能缺失,Windows Server操作系统 IIS添加WCF功能

    注: Silverlight工作流控件,如果在网页中打不开,则要判断是否缺少Silverlight工作流控件的插件程序,如果不是则可以进行一下步骤检查,如果是以下原因则是:由于IIS版本问题,安装后可 ...

  4. 如何实现图像搜索,文搜图,图搜图,CLIP+faiss向量数据库实现图像高效搜索

    如何实现图像搜索,文搜图,图搜图,CLIP+faiss向量数据库实现图像高效搜索 这是AIGC的时代,各种GPT大模型生成文本,还有多模态图文并茂大模型, 以及stable diffusion和sta ...

  5. 【UniApp】-uni-app-修改组件主题和样式

    前言 好,经过上个章节的介绍完毕之后,了解了一下 uni-app-扩展组件 那么了解完了uni-app-扩展组件之后,这篇文章来给大家介绍一下 uni-app-修改组件主题和样式 首先不管三七二十一, ...

  6. Filter拦截问题

    关于Filter拦截问题 刚开始我创建了个servlet项目一直拦截不成功 首先是因为导包的问题      import javax.servlet.*; 必须是这个包才有 第一个这个Javax.se ...

  7. C++学习笔记一:windows系统配置C++开发环境(VS code+g++/clang++)

    1.下载vscode 官网下载地址: https://code.visualstudio.com/ 安装时选择把软件加入到环境变量中这个选项 2.打开vscode,安装c/c++扩展插件 3.下载gc ...

  8. Oracle-lsnrctl监听进程控制

    LSNRCTL> help The following operations are available An asterisk (*) denotes a modifier or extend ...

  9. Volcano 原理、源码分析(二)

    0. 总结前置 1. 概述 2. 寻找调度器中的 PodGroup 2.1 从 PodGroup 到 JobInfo 的封装 2.2 从 Pod 到 TaskInfo 的封装 3. 控制器中 PodG ...

  10. 技本功|Hive优化之监控(三)

    Hive是大数据领域常用的组件之一,主要是大数据离线数仓的运算,关于Hive的性能调优在日常工作和面试中是经常涉及的一个点,因此掌握一些Hive调优是必不可少的技能.影响Hive效率的主要有数据倾斜. ...