当前位置: 首页 > article >正文

线性DP 区间DP C++

线性DP

题一 数字三角形

图源ACWING

解题思路

在这里插入图片描述
三角形内的某个点,可以从这个点的左上方或右上方来到这个点,因此有状态转移方程:
f[i, j] = max(f[i - 1, j - 1] + a[i][j], f[i - 1][j] + a[i][j])

代码实现

#include<iostream>
#include<algorithm>
#include<cstring>

using namespace std;

const int N = 510;

int f[N][N], a[N][N];

int main()
{
    int n;
    cin >> n;
    
    for (int i = 1; i <= n; i ++ )
    {
        for (int j = 1; j <= i; j ++ )
        {
            scanf("%d", &a[i][j]);
        }
    }
    
    memset(f, -0x3f, sizeof f);//因为有负数的存在,所以需要初始化为负无穷,方便比大小取值
    
    f[1][1] = a[1][1];
    
    for (int i = 2; i <= n; i ++ )
    {
        for (int j = 1; j <= i; j ++ )
        {
            f[i][j] = max(f[i - 1][j - 1] + a[i][j], f[i - 1][j] + a[i][j]);
        }
    }
    
    int res = -1e9;
    for (int j = 1; j <= n; j ++ )
    {
        res = max(res, f[n][j]);
    }
    
    cout << res;
    return 0;
}

题二 最长上升子序列

图源ACWING

解题思路

假设题目给出了n个数,这n个数存在了a[N]中:
其中f[i] 表示以a[i]为结尾的最长上升子序列的长度(位置也是独特的,假设a[N]中有多个相同的数,他们的f值也不一定相同!!),则如果现在在计算第i个数的f[i]值,且遍历到了第j个数(j <= i) 并且 a[i] > a[]则有f[i] = max(f[i], f[j] + 1);

图源ACWING
PS:最好在纸上模拟一次下面代码中的实现, 能很好地理解这个思路

代码实现

#include<iostream>
#include<algorithm>

using namespace std;

const int N = 1010;

int f[N], a[N];

int main()
{
    int n;
    cin >> n;
    
    for (int i = 1; i <= n; i ++ )
    {
        scanf("%d", &a[i]);
    }
    
    for (int i = 1; i <= n; i ++ )
    {
        f[i] = 1;//第i个数的f值至少为1, 因为这个上升序列至少包含第i个数本身
        for (int j = 1; j <= i; j ++ )
        {
            if (a[i] > a[j])
            {
                f[i] = max(f[i], f[j] + 1);
            }
        }
    }
    
    int res = -0x3f3f3f3f;
    for (int i = 1; i <= n; i ++ )
    {
        res = max(res, f[i]);
    }
    
    cout << res;
    return 0;
}

区间DP

题目

在这里插入图片描述

解题思路

在这里插入图片描述
i是左端点, j是右端点, k是分界线,s[i]是石子堆的前缀和
当最后一步将从i到k的石子与从k + 1到j的石子合并时, 其代价为s[j] - s[i - 1](前缀和)

代码实现

#include<iostream>
#include<algorithm>
#include<cstring>

using namespace std;

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

int main()
{
    int n;
    cin >> n;
    
    for (int i = 1; i <= n; i ++ )
    {
        scanf("%d", &s[i]);
        s[i] += s[i - 1];
    }
    
    for (int len = 2; len <= n; len ++ )
    {
        for (int i = 1; i + len - 1; i ++ )
        {
            int l = i, r = i + len - 1;
            f[l][r] = 0x3f3f3f3f;
            for (int k = i; 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];
    return 0;
}

http://www.kler.cn/a/394655.html

相关文章:

  • 微服务day08
  • 深入探索:Scrapy深度爬取策略与实践
  • 网络技术-定义配置ACL规则的语法和命令
  • C++算法练习-day40——617.合并二叉树
  • Unity 性能优化方案
  • C# 模拟浏览器自操作(自动化办公)
  • Cyberchef配合Wireshark提取并解析HTTP/TLS流量数据包中的文件
  • Python中的正则表达式教程
  • 正则表达式那些事儿
  • 融合创新:CNN+LSTM在深度学习中的高效应用,助力科研发表高影响因子文章!
  • Linux之文件和目录类命令详解(2)
  • 在 Windows 11 中使用 MuMu 模拟器 12 国际版配置代理
  • Unity3D高级编程
  • 离线语音识别自定义功能怎么用?
  • C#预处理器指令#if和#endif:掌握条件编译的艺术
  • 使用 Vision 插件让 GitHub Copilot 识图问答
  • windows C#-异常处理
  • 中断的硬件框架
  • 贪心算法day 06
  • Docker 中启动 NGINX 并配置 HTTPS 443 端口
  • 如何用Java爬虫“偷窥”淘宝商品类目API的返回值
  • Linux学习,ip 命令
  • 介绍一下位操作符(c基础)
  • python调用MySql详细步骤
  • 【干货】仓储管理SOP标准化操作!
  • torchvision库在进行图片转换操作中报antialias参数没有显式设置会导致不同图片后端中的值不统一的警告信息