acwing算法基础之动态规划--线性DP和区间DP

这篇具有很好参考价值的文章主要介绍了acwing算法基础之动态规划--线性DP和区间DP。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1 基础知识

线性DP:状态转移表达式存在明显的线性关系。
区间DP:与顺序有关,状态与区间有关。

2 模板

3 工程化

题目1:数字三角形。

解题思路:直接DP即可,f[i][j]可以来自f[i-1][j] + a[i][j]f[i-1][j-1] + a[i][j],注意f[i-1][j]不存在的情况(最后一个点)和f[i-1][j-1]不存在的情况(第一个点)。

C++代码如下,文章来源地址https://www.toymoban.com/news/detail-757165.html

#include <iostream>

using namespace std;

const int N = 510;
int n;
int a[N][N];
int f[N][N];

int main() {
    cin >> n;
    for (int i = 0; i < n; ++i) {
        for (int j = 0; j < i + 1; ++j) {
            cin >> a[i][j];
        }
    }
    
    for (int i = 0; i < N; ++i) {
        for (int j = 0; j < N; ++j) {
            f[i][j] = -0x3f3f3f3f;
        }
    }
    
    f[0][0] = a[0][0];
    for (int i = 1; i < n; ++i) {
        for (int j = 0; j < i + 1; ++j) {
            f[i][j] = max(f[i][j], f[i-1][j] + a[i][j]);
            if (j - 1 >= 0) f[i][j] = max(f[i][j], f[i-1][j-1] + a[i][j]);
        }
    }
    
    int res = -0x3f3f3f3f;
    for (int j = 0; j < n; ++j) res = max(res, f[n-1][j]);
    
    cout << res << endl;
    
    return 0;
}

题目2:最长上升子序列。注意可以连续,比如3 1 2 1 8 5 6,那么它的最长上升子序列为1 2 5 6,长度为4。

解题思路:DP即可。

状态定义,f[i]表示以第 i i i元素结尾的上升子序列的最大长度。

状态转移,有,

  1. 没有上一个元素,即f[i] = 1
  2. 从第 i − 1 i-1 i1个元素转移过来,需要满足a[i-1] < a[i],则f[i-1] + 1
  3. 从第 i − 2 i-2 i2个元素转移过来,需要满足a[i-2] < a[i],则f[i-2] + 1
    ……
  4. 从第 0 0 0个元素转移过来,需要满足a[0] < a[i],则f[0] + 1

C++代码如下,

#include <iostream>

using namespace std;

const int N = 1010;
int n;
int a[N];
int f[N];

int main() {
    cin >> n;
    for (int i = 0; i < n; ++i) cin >> a[i];
    
    for (int i = 0; i < n; ++i) {
        f[i] = 1;
        for (int j = 0; j < i; ++j) {
            if (a[j] < a[i]) f[i] = max(f[i], f[j] + 1);
        } 
    }
    
    int res = 0;
    for (int i = 0; i < n; ++i) res = max(res, f[i]);
    
    cout << res << endl;
    
    return 0;
}

同时输出最长上升子序列,那么此时只需要记住当前状态是从哪一个状态转移过来的,然后逆序输出即可。C++代码如下,

#include <iostream>
#include <algorithm>

using namespace std;

const int N = 1010;
int n;
int a[N];
int f[N];
int g[N];

int main() {
    cin >> n;
    for (int i = 0; i < n; ++i) cin >> a[i];
    
    for (int i = 0; i < n; ++i) {
        f[i] = 1;
        g[i] = -1; //g[i]=-1表示它是子序列的起点
        for (int j = 0; j < i; ++j) {
            if (a[j] < a[i]) {
                if (f[i] < f[j] + 1) {
                    f[i] = f[j] + 1;
                    g[i] = j;//g[i]=j表示状态f[i]是从状态f[j]转移过来的
                }
            }
        } 
    }
    
    int res = 0;
    for (int i = 0; i < n; ++i) {
        if (f[i] > f[res]) {
            res = i;
        }
    }
    
    cout << f[res] << endl;//输出最长子序列的长度
    
    vector<int> ans; //最长子序列
    for (int i = res; i != -1; i = g[i]) {
        ans.emplace_back(a[i]);
    }
    reverse(ans.begin(), ans.end());
    
    for (int i = 0; i < ans.size(); ++i) cout << ans[i] << " ";
    cout << endl;
    
    return 0;
}

测试样例,

输入:
7
3 1 2 1 8 5 6
输出:
4
1 2 5 6 

题目3:最长公共子序列,给定字符串a和字符串b,求它们的最长公共子序列。例如acbdabedc,则它们的最长公共子序列为abd,长度为3。

思路:利用DP来求解,由于这里求的是最大长度,因此在状态转移时,可以适当放大,以便可以表示相应转移路径。

状态定义,f[i][j]:从字符串a前i个中选且从字符串b中前j个中选的公共子序列的最大长度。

状态转移,有,

  1. a[i]不在最优公共子序列当中,b[j]不在最优公共子序列当中,则f[i-1][j-1]
  2. a[i]在最优公共子序列当中,b[j]不在最优公共子序列当中,无法表示???,故将其放大,为f[i][j-1]
  3. a[i]不在最优公共子序列当中,b[j]在最优公共子序列当中,无法表示???,故将其放大,为f[i-1][j]
  4. a[i]在最优公共子序列当中,b[j]在最优公共子序列当中,要求a[i]=b[j],则f[i-1][j-1] + 1

进一步思考,由于进行了放大,因此第(2)种情况和第(3)种情况包含了第(1)种情况。因此在代码实现时,可以考虑如下状态转移f[i][j-1]f[i-1][j]f[i-1][j-1] + 1

C++代码如下,

#include <iostream>

using namespace std;

const int N = 1010;
char a[N], b[N];
int n, m;
int f[N][N];

int main() {
    cin >> n >> m;
    cin >> a + 1 >> b + 1;
    
    for (int i = 1; i <= n; ++i) {
        for (int j = 1; j <= m; ++j) {
            f[i][j] = max(f[i-1][j], f[i][j-1]);
            if (a[i] == b[j]) f[i][j] = max(f[i][j], f[i-1][j-1] + 1);
        }
    }
    
    cout << f[n][m] << endl;
    
    return 0;
}

题目4:合并相邻石子的最小代价。例如1 3 5 2,合并前两堆石子13的代价是4,得到4 5 2

解题思路:区间合并类DP,状态定义围绕着区间,然后怎么层层递进计算状态,比较考验。答案是枚举区间长度来计算。

状态定义,f[i][j]:合并第i~j堆石子的最小代价。

状态转移,有,

  1. 最后一次合并为[i,i]和[i+1,j]这两个区间合并,即f[i][i] + f[i + 1][j] + s[j] - s[i - 1],其中s[i]表示前缀和。
  2. 最后一次合并为[i,i+1]和[i+2,j]这两个区间合并,即f[i][i + 1] + f[i + 2][j] + s[j] - s[i - 1]
  3. 最后一次合并为[i,i+2]和[i+3,j]这两个区间合并,即f[i][i + 2] + f[i + 3][j] + s[j] - s[i - 1]
    ……
  4. 最后一次合并为[i,j-1]和[j,j]这两个区间合并,即f[i][j - 1] + f[j][j] + s[j] - s[i - 1]

初始化,f[i][i]=0,然后其余初始化为正无穷大,比如1e9

最终答案为,f[1][n]

注意状态的遍历,是通过区间长度从1到n实现的。

C++代码如下,

#include <iostream>

using namespace std;

const int N = 310;
int a[N];
int s[N];
int f[N][N];
int n;

int main() {
    cin >> n;
    for (int i = 1; i <= n; ++i) cin >> a[i];
    
    for (int i = 1; i <= n; ++i) s[i] += s[i-1] + a[i];
    
    for (int len = 2; len <= n; ++len) {
        //枚举左端点
        for (int i = 0; i + len - 1 <= n; ++i) {
            int l = i, r = i + len - 1;
            f[l][r] = 1e9;
            for (int k = l; k < r; ++k) {
                f[l][r] = min(f[l][r], f[l][k] + f[k+1][r] + s[r] - s[l-1]);
            }
        }
    }
    
    cout << f[1][n] << endl;
    
    return 0;
}

到了这里,关于acwing算法基础之动态规划--线性DP和区间DP的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

  • 算法基础复盘笔记Day10【动态规划】—— 线性DP

    ❤ 作者主页:欢迎来到我的技术博客😎 ❀ 个人介绍:大家好,本人热衷于 Java后端开发 ,欢迎来交流学习哦!( ̄▽ ̄)~* 🍊 如果文章对您有帮助,记得 关注 、 点赞 、 收藏 、 评论 ⭐️⭐️⭐️ 📣 您的支持将是我创作的动力,让我们一起加油进步吧!!!🎉🎉 1. 题目

    2023年04月21日
    浏览(65)
  • C++动态规划-线性dp算法

    莫愁千里路 自有到来风 CSDN 请求进入专栏                                    X 是否进入《 C++ 专栏》? 确定 目录  线性dp简介 斐波那契数列模型  第N个泰波那契数 思路: 代码测试:  三步问题 思路: 代码测试: 最小花费爬楼梯 思路: 代码测试:  路径问题 数字三

    2024年02月19日
    浏览(35)
  • 猿创征文 |【算法面试入门必刷】动态规划-线性dp(一)

    📦个人主页:一二三o-0-O的博客 🏆技术方向:C/C++客户端资深工程师(直播+音视频剪辑) 👨‍💻作者简介:数据结构算法与音视频领域创作者 📒 系列专栏:牛客网面试必刷 📣专栏目标:帮助伙伴们通过系统训练,掌握数据结构与算法,收获心仪Offer 📝推荐一个找工作

    2024年02月03日
    浏览(35)
  • 猿创征文 |【算法面试入门必刷】动态规划-线性dp(四)

    📦个人主页:一二三o-0-O的博客 🏆技术方向:C/C++客户端资深工程师(直播+音视频剪辑) 👨‍💻作者简介:数据结构算法与音视频领域创作者 📒 系列专栏:牛客网面试必刷 📣专栏目标:帮助伙伴们通过系统训练,掌握数据结构与算法,收获心仪Offer 📝推荐一个找工作

    2024年02月02日
    浏览(27)
  • acwing算法基础之动态规划--背包问题

    (零) 背包问题描述:有 N N N 个物品,每个物品的体积是 v i v_i v i ​ ,价值是 w i w_i w i ​ ,现有容量是 V V V 的背包,求这个背包能装下的物品的最大价值。 01背包问题:每个物品只有1个。 完全背包问题:每个物品有无穷多个。 多重背包问题:第 i i i 个物品有 s i s_i s

    2024年02月05日
    浏览(35)
  • 区间dp(动态规划)

    动态规划(dp)是一种通过将问题分解为子问题,并利用已解决的子问题的解来求解原问题的方法。适用于具有重叠子问题和最优子结构性质的优化问题。通过定义状态和状态转移方程,动态规划可以在避免重复计算的同时找到问题的最优解,是一种高效的求解方法,常用于

    2024年02月15日
    浏览(28)
  • 动态规划——区间DP 学习笔记

    不含四边形不等式优化。 线性动态规划的局限性在于,它只能顺推或倒退,而不能有子区间依赖的问题。 区间动态规划是线性动态规划的扩展,它将问题划分为若干个子区间,并通过定义状态和状态转移方程来求解每个子区间的最优解,最终得到整个区间的最优解。 区间动

    2024年02月08日
    浏览(31)
  • 动态规划——区间dp [石子合并]

    动态规划(dp)是一种通过将问题分解为子问题,并利用已解决的子问题的解来求解原问题的方法。适用于具有重叠子问题和最优子结构性质的优化问题。通过定义状态和状态转移方程,动态规划可以在避免重复计算的同时找到问题的最优解,是一种高效的求解方法,常用于

    2024年02月12日
    浏览(36)
  • Acwing-基础算法课笔记之动态规划(背包问题)

    01背包中的0和1指的是放与不放,而且不能出现放多个的情况,背包只能放相同物品中的一个; 首先是对 d [ i ] [ j ] d[i][j] d [ i ] [ j ] 数组的解释,该数组表示的是只看前 i i i 个物品,总体积是 j j j 的情况下,总价值最大是多少; 不选第 i i i 个物品,只考虑前 i − 1 i-1 i −

    2024年04月17日
    浏览(37)
  • 动态规划系列 | 一文搞定区间DP

    区间 DP 可以用于解决一些涉及到区间合并或分割的问题。区间 DP 通常有以下三个特点: 合并(分割) :将两个或多个部分进行整合,或者反过来将一个区间分解成多个部分。 特征 :能将问题分解为能两两合并的形式。 求解 :对整个问题设最优解,枚举合并点,将问题分

    2024年02月02日
    浏览(32)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包