石子合并-环(区间dp)c++

2024-08-31 18:44
文章标签 c++ dp 合并 区间 石子

本文主要是介绍石子合并-环(区间dp)c++,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

刚才的《石子合并》问题是把石子放成一排,如果石子放成一个环,第 N 堆和第 1 堆相邻,又该怎么做呢?

我们要想办法把它变成之前放成一排的问题,可以发现只要我们把

  • a1​,a2​,a3​,⋯,aN​
  • a2​,a3​,a4​,⋯,aN​,a1​
  • a3​,a4​,a5​,⋯,aN​,a1​,a2​
  • aN​,a1​,a2​,⋯,aN−1​

这 N 种放成一排的情况都考虑清楚,取其中的最优解,就一定是放成环的样子的最优解。

但如果做 N 次区间 DP 时间复杂度就变成了 O(n4),并且我们可以发现在其中有一些状态是重复计算的。

有一种很常用的方法可以避免这样的重复计算,就是把数组复制一遍,把复制的放在原数组的后边,形成一个新的序列 b,其中的元素是:

a1​,a2​,a3​,⋯,aN​,a1​,a2​,a3​,⋯,aN​

此时,

  • a1​,a2​,a3​,⋯,aN​
  • 1a2​,a3​,a4​,⋯,aN​,a1​
  • a3​,a4​,a5​,⋯,aN​,a1​,a2​
  • aN​,a1​,a2​,⋯,aN−1​

这里的每一种情况后对应成了复制完成后数组的一段连续区间。

  • a1​,a2​,a3​,⋯,aN​ 是区间b1​∼bN​
  • a2​,a3​,a4​,⋯,aN​,a1​ 是区间 b2​∼bN+1​
  • a3​,a4​,a5​,⋯,aN​,a1​,a2​ 是区间 b3​∼bN+2​
  • aN​,a1​,a2​,⋯,aN−1​ 是区间bN​∼b2×N−1​

所以我们只需要正常做区间 DP ,最后求dp[1][N],dp[2][N+1],…,dp[N][2×N−1] 的最小值即可。

#include <iostream>
#include <algorithm>
#include <cstring>
using namespace std;
const int inf = 0x3f3f3f3f;
int a[205], sum[205], dp[205][205], ans = inf;
int main() {int n;cin >> n;for (int i = 1; i <= n; i++) {cin >> a[i];a[i + n] = a[i];}memset(dp, 0x3f, sizeof(dp));for (int i = 1; i <= 2 * n; i++) {sum[i] = sum[i - 1] + a[i];dp[i][i] = 0;}for (int len = 2; len <= n; len++) {for (int i = 1; i <= n; i++) {int j = i + len - 1;for (int k = i; k < j; k++) {dp[i][j] = min(dp[i][j], dp[i][k] + dp[k + 1][j] + sum[j] - sum[i - 1]);}}}for (int i = 1; i <= n; i++) {ans = min(ans, dp[i][i + n - 1]);}cout << ans;return 0;
}

 

这篇关于石子合并-环(区间dp)c++的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1124702

相关文章

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

C++变换迭代器使用方法小结

《C++变换迭代器使用方法小结》本文主要介绍了C++变换迭代器使用方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、源码2、代码解析代码解析:transform_iterator1. transform_iterat

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.

C++中std::distance使用方法示例

《C++中std::distance使用方法示例》std::distance是C++标准库中的一个函数,用于计算两个迭代器之间的距离,本文主要介绍了C++中std::distance使用方法示例,具... 目录语法使用方式解释示例输出:其他说明:总结std::distance&n编程bsp;是 C++ 标准

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示