石子合并+环形石子合并+能量项链+凸多边形的划分——区间DP

这篇具有很好参考价值的文章主要介绍了石子合并+环形石子合并+能量项链+凸多边形的划分——区间DP。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、石子合并 (经典例题)

设有 N 堆石子排成一排,其编号为 1,2,3,…,N。
每堆石子有一定的质量,可以用一个整数来描述,现在要将这 N 堆石子合并成为一堆。
每次只能合并相邻的两堆,合并的代价为这两堆石子的质量之和,合并后与这两堆石子相邻的石子将和新堆相邻,合并时由于选择的顺序不同,合并的总代价也不相同。
例如有 4 堆石子分别为 1 3 5 2, 我们可以先合并 1、2 堆,代价为 4,得到 4 5 2, 又合并 1、2 堆,代价为 9,得到 9 2 ,再合并得到 11,总代价为 4+9+11=24;
如果第二步是先合并 2、3 堆,则代价为 7,得到 4 7,最后一次合并代价为 11,总代价为 4+7+11=22。

问题是:找出一种合理的方法,使总的代价最小,输出最小代价。

输入
第一行一个数 N 表示石子的堆数 N (1 ≤ N ≤ 300)。
第二行 N 个数,表示每堆石子的质量(均不超过 1000)。

输出
输出一个整数,表示最小代价。

Input
4
1 3 5 2

Output
22
解析:
用一个状态表示一个区间,f[i][j] 表示将第 i 堆石子到第 j 堆石子合并成一堆石子的合并方式的集合。
状态转移:
例如,将 区间 [i,j] 分为 [i,k] 和 [k+1,j] ,枚举 k 在区间[i,j]的位置就能找到最小代价使得 合并这两个区间的代价最小,即 f[i][k]+f[k+1,j] 的值最小,再加上一个第 i 堆到第 j 堆的总重量即可。
因为在计算状态的时候,得保证在这个状态之前的状态已经计算完毕,所以可以采用循环区间长度从小到大的方式,进行计算。

#include <bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define ios ios::sync_with_stdio(false),cin.tie(nullptr),cout.tie(nullptr);
int gcd(int a,int b) { return b? gcd(b,a%b) : a; }
typedef pair<int,int> PII;
const double PI=acos(-1.0);
const int N=310;
int n;
int s[N];
int f[N][N];
void solve()
{
    cin>>n;
    for (int i=1;i<=n;i++) cin>>s[i];
    for (int i=1;i<=n;i++) s[i] +=s[i-1];
    
    for (int len=1;len<=n;len++)
    for (int i=1;i+len-1<=n;i++)
    {
        int l=i,r=i+len-1;
        if(len!=1) f[l][r]=2e9;                             //因为当len=1时,只有一堆石子,不需要合并,代价为0
        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];
}
signed main()
{
    ios;
    int T=1;
    //cin>>T;
    while (T--) solve();
    return 0;
}

 

二、 环形石子合并 (加个环)

将 n 堆石子绕圆形操场排放,现要将石子有序地合并成一堆。
规定每次只能选相邻的两堆合并成新的一堆,并将新的一堆的石子数记做该次合并的得分。
请编写一个程序,读入堆数 n 及每堆的石子数,并进行如下计算:
选择一种合并石子的方案,使得做 n−1 次合并得分总和最大。
选择一种合并石子的方案,使得做 n−1 次合并得分总和最小。

输入
第一行包含整数 n (1 ≤ n ≤ 200),表示共有 n 堆石子。
第二行包含 n 个整数,分别表示每堆石子的数量。

输出
共两行,第一行为合并得分总和最小值,第二行为合并得分总和最大值。

Input
4
4 5 9 4

Output
43
54

解析:
与上一题的石子合并的想法基本相似,不同的是这道题的石堆的摆放方式是环式的。
在上道题的基础上,将这个环断开,变成一条链,枚举每个断点。
不过直接枚举的话,时间复杂度是 n^4,超时了。
我们可以将数组读入,再将数组复制一遍,再接到这个数组上。
这样时间复杂度就降下来了,时间复杂度在 n^3。
注意:在做这种环式的题,就可以多想一想这样的方法

#include <bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define ios ios::sync_with_stdio(false),cin.tie(nullptr),cout.tie(nullptr);
int gcd(int a,int b) { return b? gcd(b,a%b) : a; }
typedef pair<int,int> PII;
const double PI=acos(-1.0);
const int N=410;
int n;
int a[N],s[N];
int f[N][N],g[N][N];
void solve()
{
    cin>>n;
    for (int i=1;i<=n;i++) cin>>a[i],a[i+n]=a[i];
    for (int i=1;i<=2*n;i++) s[i]=s[i-1]+a[i];
    
    for (int len=1;len<=n;len++)                //一层循环枚举区间长度
    for (int i=1;i+len-1<=2*n;i++)              //二层循环枚举区间的左右端点
    {
        int l=i,r=i+len-1;
        if (len!=1) f[l][r]=2e9,g[l][r]=-2e9;
        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]);
            g[l][r]=max(g[l][r],g[l][k]+g[k+1][r]+s[r]-s[l-1]);
        }
    }
    
    int minx=2e9,maxx=-2e9;                     //找到最小值,最大值
    for (int i=1;i<=n;i++) 
    {
        minx=min(minx,f[i][i+n-1]);
        maxx=max(maxx,g[i][i+n-1]);
    }
    
    cout<<minx<<endl<<maxx<<endl;
}
signed main()
{
    ios;
    int T=1;
    //cin>>T;
    while (T--) solve();
    return 0;
}

三、能量项链 (跟上个题一样)

在 Mars 星球上,每个 Mars 人都随身佩带着一串能量项链,在项链上有 N 颗能量珠。
能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数。
并且,对于相邻的两颗珠子,前一颗珠子的尾标记一定等于后一颗珠子的头标记。
因为只有这样,通过吸盘(吸盘是 Mars 人吸收能量的一种器官)的作用,这两颗珠子才能聚合成一颗珠子,同时释放出可以被吸盘吸收的能量。

如果前一颗能量珠的头标记为 m,尾标记为 r,后一颗能量珠的头标记为 r,尾标记为 n,则聚合后释放的能量为 m×r×n(Mars 单位),新产生的珠子的头标记为 m,尾标记为 n。
需要时,Mars 人就用吸盘夹住相邻的两颗珠子,通过聚合得到能量,直到项链上只剩下一颗珠子为止。
显然,不同的聚合顺序得到的总能量是不同的,请你设计一个聚合顺序,使一串项链释放出的总能量最大。

例如:设 N=4,4 颗珠子的头标记与尾标记依次为 (2,3)(3,5)(5,10)(10,2)。
我们用记号 ⊕ 表示两颗珠子的聚合操作,(j⊕k) 表示第 j,k 两颗珠子聚合后所释放的能量。则
第 4、1 两颗珠子聚合后释放的能量为:(4⊕1)=10×2×3=60。
这一串项链可以得到最优值的一个聚合顺序所释放的总能量为 ((4⊕1)⊕2)⊕3)=10×2×3+10×3×5+10×5×10=710。

输入
输入的第一行是一个正整数 N (4 ≤ N ≤ 100),表示项链上珠子的个数。
第二行是 N 个用空格隔开的正整数,所有的数均不超过 1000,第 i 个数为第 i 颗珠子的头标记,当 i<N 时,第 i 颗珠子的尾标记应该等于第 i+1 颗珠子的头标记,第 N 颗珠子的尾标记应该等于第 1 颗珠子的头标记。
至于珠子的顺序,你可以这样确定:将项链放到桌面上,不要出现交叉,随意指定第一颗珠子,然后按顺时针方向确定其他珠子的顺序。

输出
输出只有一行,是一个正整数 E (1 ≤ E ≤ 2e9),为一个最优聚合顺序所释放的总能量。

Input
4
2 3 5 10

Output
710

解析:
跟上一题一样,纯套路。

//代码一:开个结构体存储头和尾,相当于一个点,就跟上道题一模一样了
#include <bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define ios ios::sync_with_stdio(false),cin.tie(nullptr),cout.tie(nullptr);
int gcd(int a,int b) { return b? gcd(b,a%b) : a; }
typedef pair<int,int> PII;
const double PI=acos(-1.0);
const int N=400;
struct node
{
    int x,y;
}str[N];
int n;
int a[N];
int f[N][N];
void solve()
{
    cin>>n;
    for (int i=1;i<=n;i++) cin>>a[i];
    for (int i=1;i<=n;i++)
    {
        if (i==1)  str[i]={a[n],a[i]};
        else str[i]={a[i-1],a[i]};
        str[i+n]=str[i];
    }
    
    for (int len=1;len<=n;len++)
    for (int i=1;i+len-1<=2*n;i++)
    {
        int l=i,r=i+len-1;
        if (len!=1) f[l][r]=-2e9;
        for (int k=l;k<r;k++)
        {
            f[l][r]=max(f[l][r],f[l][k]+f[k+1][r]+str[l].x*str[k].y*str[r].y);          //左区间左端点的头*左区间右端点的尾*右区间右端点的尾
        }
    }
    
    int ans=-2e9;
    for (int i=1;i<=n;i++) ans=max(ans,f[i][i+n-1]);
    cout<<ans;
}
signed main()
{
    ios;
    int T=1;
    //cin>>T; 
    while (T--) solve();
    return 0;
}


//代码二
//划分方式:f(l,r)=f(l,k)+f(k,r)+a[l]*a[k]*a[r]
#include <bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define ios ios::sync_with_stdio(false),cin.tie(nullptr),cout.tie(nullptr);
int gcd(int a,int b) { return b? gcd(b,a%b) : a; }
typedef pair<int,int> PII;
const double PI=acos(-1.0);
const int N=400;
int n;
int a[N];
int f[N][N];
void solve()
{
    cin>>n;
    for (int i=1;i<=n;i++) cin>>a[i],a[i+n]=a[i];
    
    for (int len=3;len<=n+1;len++)                      //至少 3 个点
    for (int i=1;i+len-1<=2*n;i++)
    {
        int l=i,r=i+len-1;
        f[l][r]=-2e9;
        for (int k=l;k<r;k++)
        {
            f[l][r]=max(f[l][r],f[l][k]+f[k][r]+a[l]*a[k]*a[r]); 
        }
    }
    
    int ans=-2e9;
    for (int i=1;i<=n;i++) ans=max(ans,f[i][i+n]);      //查找区间长度为 n+1 的
    cout<<ans;
}
signed main()
{
    ios;
    int T=1;
    //cin>>T; 
    while (T--) solve();
    return 0;
}

 四、凸多边形的划分

给定一个具有 N 个顶点的凸多边形,将顶点从 1 至 N 标号,每个顶点的权值都是一个正整数。
将这个凸多边形划分成 N−2 个互不相交的三角形,对于每个三角形,其三个顶点的权值相乘都可得到一个权值乘积,试求所有三角形的顶点权值乘积之和至少为多少。

输入
第一行包含整数 N (N ≤ 50),表示顶点数量。
第二行包含 N 个整数,依次为顶点 1 至顶点 N 的权值。

输出
输出仅一行,为所有三角形的顶点权值乘积之和的最小值。
数据保证所有顶点的权值都小于 1e9

Input
5
121 122 123 245 231

Output
12214884

 解析:

石子合并+环形石子合并+能量项链+凸多边形的划分——区间DP,算法,数据结构
f[l][r] 表示 所有 由(l,l+1)(l+1,l+2)……(r-1,r)(r,l)这些边组成的多边形 划分成三角形的方案的集合。
跟上道题的划分是一样的,不过不需要破环成链,因为枚举哪一条边都是一样的。
这里的运算数据太大,需要高精度运算。

//代码一:没有加高精度的代码
#include <bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define ios ios::sync_with_stdio(false),cin.tie(nullptr),cout.tie(nullptr);
int gcd(int a,int b) { return b? gcd(b,a%b) : a; }
typedef pair<int,int> PII;
const double PI=acos(-1.0);
const int N=110;
int n;
int w[N];
int f[N][N];
int INF=1e18;
void solve()
{
    cin>>n;
    for (int i=1;i<=n;i++) cin>>w[i];
    
    for (int len=3;len<=n;len++)
    for (int i=1;i+len-1<=n;i++)
    {
        int l=i,r=i+len-1;
        f[l][r]=INF;
        for (int k=l;k<r;k++)
        {
            f[l][r]=min(f[l][r],f[l][k]+f[k][r]+w[l]*w[k]*w[r]);
        }
    }
    
    cout<<f[1][n];
}
signed main()
{
    ios;
    int T=1;
    //cin>>T; 
    while (T--) solve();
    return 0;
}



//代码二:加了高精度的代码
#include <bits/stdc++.h>
using namespace std;
#define int long long
#define endl '\n'
#define ios ios::sync_with_stdio(false),cin.tie(nullptr),cout.tie(nullptr);
int gcd(int a,int b) { return b? gcd(b,a%b) : a; }
typedef pair<int,int> PII;
const double PI=acos(-1.0);
const int N=55,M=35,INF=1e9;
int n;
int w[N];
int f[N][N][M];
int c[M];
void add(int a[],int b[])
{
    memset(c,0,sizeof c);
    int t=0;
    for (int i=0;i<M;i++)
    {
        t +=a[i]+b[i];
        c[i]=t%10;
        t /=10;
    }
    memcpy(a,c,sizeof c);
}
void mul(int a[],int b)
{
    memset(c,0,sizeof c);
    int t=0;
    for (int i=0;i<M;i++)
    {
        t +=a[i]*b;
        c[i]=t%10;
        t /=10;
    }
    memcpy(a,c,sizeof c);
}
int cmp(int a[],int b[])
{
    for (int i=M-1;i>=0;i--)
    {
        if (a[i]>b[i]) return 1;
        else if (a[i]<b[i]) return -1;
    }
    return 0;
}
void print(int a[])
{
    int k=M-1;
    while (k&&!a[k]) k--;
    while (k>=0) cout<<a[k--];
    cout<<endl;
}
int temp[M];
void solve()
{
    cin>>n;
    for (int i=1;i<=n;i++) cin>>w[i];
    
    for (int len=3;len<=n;len++)
    for (int i=1;i+len-1<=n;i++)
    {
        int l=i,r=i+len-1;
        f[l][r][M-1]=1;
        for (int k=l+1;k<r;k++)
        {
            memset(temp,0,sizeof temp);
            temp[0]=w[l];
            mul(temp,w[k]);
            mul(temp,w[r]);
            add(temp,f[l][k]);
            add(temp,f[k][r]);
            if (cmp(f[l][r],temp)>0) memcpy(f[l][r],temp,sizeof temp);
        }
    }
    
    print(f[1][n]);
}
signed main()
{
    ios;
    int T=1;
    //cin>>T; 
    while (T--) solve();
    return 0;
}

 文章来源地址https://www.toymoban.com/news/detail-827916.html

到了这里,关于石子合并+环形石子合并+能量项链+凸多边形的划分——区间DP的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

  • 石子合并一章通(环形石子合并,四边形不等式,GarsiaWachs算法)(内附封面)

    在一个圆形操场的四周摆放 N N N 堆石子,现要将石子有次序地合并成一堆,规定每次只能选相邻的 2 2 2 堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分。 试设计出一个算法,计算出将 N N N 堆石子合并成 1 1 1 堆的最小得分和最大得分。 数据的第 1 1 1 行是正

    2024年02月14日
    浏览(28)
  • 石子合并系列问题

    石子合并问题在网上有三个版本: AcWing 282. 石子合并 设有 N 堆石子排成一排,其编号为 1,2,3,…,N。 每堆石子有一定的质量,可以用一个整数来描述,现在要将这 N 堆石子合并成为一堆。 每次只能合并相邻的两堆,合并的代价为这两堆石子的质量之和,合并后与这两堆

    2024年02月02日
    浏览(47)
  • 石子合并(区间dp)

    (1)f[l][r]表示l~r之间合并的最小代价。 (2)将l~r拆成l~k,k+1~r两份分别最优合并,再把两份合并,对于每个l,r通过枚举所有可能k探寻最优。 (3)最终目标是f[1][n],注意到长区间是由短区间递推出来的,所以由小到大枚举区间长度,再枚举起点,此时l = 起点,r = 起点 +len

    2024年02月10日
    浏览(26)
  • 石子合并问题(动态规划)

    石子合并问题是一个经典的动态规划问题,应用了最优子结构和重复子问题的思想。 (1)有N堆石子,现要将石子有序的合并成一堆,规定如下:每次只能移动任意的2堆石子合并,合并花费为新合成的一堆石子的数量。求将这N堆石子合并成 (动态规划)O(n^3) 设dp[i][j]表示将i至j之

    2024年02月06日
    浏览(36)
  • 合并石子(动态规划)

    首先理解题意,笔者看到这道题时首先想到的是贪心,仔细分析后发现题上没说石子根据一定顺序来摆放,易证局部最优解不一定是问题最优解,没有贪心性质,不能用贪心 然后需要强调的是,该题是一个圆形结构 这里为了方便理解,我们先将其简化为线性结构 先看一个简

    2024年02月01日
    浏览(30)
  • 【动态规划】之石子合并问题

    在一个操场上一排地摆放着N堆石子。现要将石子有次序地合并成一堆。规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石子数记为该次合并的得分。请设计一个程序,计算出将N堆石子合并成一堆的最小得分和最大得分。 我们对n的取值逐步分析 当n=1时,没有进

    2024年04月28日
    浏览(27)
  • 动态规划——区间dp [石子合并]

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

    2024年02月12日
    浏览(38)
  • 石子合并(动态规划 区间DP)+详细注释

    原题链接   活动 - AcWing 题目 设有 N 堆石子排成一排,其编号为 1,2,3,…,N。 每堆石子有一定的质量,可以用一个整数来描述,现在要将这 N 堆石子合并成为一堆。 每次只能合并相邻的两堆,合并的代价为这两堆石子的质量之和,合并后与这两堆石子相邻的石子将和新堆相

    2024年02月16日
    浏览(32)
  • 租用游艇问题 石子合并问题 动态规划实验

    实验名称:                动态规划                          一、实验预习 1、实验目的 1. 理解并掌握动态规划方法的设计思想; 2. 提高应用动态规划方法解决问题和设计算法的能力; 3. 通过编程实现租用游艇问题和石子合并问题,进一步理解动态规划方

    2024年02月07日
    浏览(39)
  • C++动态规划经典案例解析之合并石子

    区间类型问题,指求一个数列中某一段区间的值,包括求和、最值等简单或复杂问题。此类问题也适用于动态规划思想。 如 前缀和 就是极简单的区间问题。如有如下数组: 现给定区间信息 [3,6] ,求区间内所有数字相加结果。即求如下图位置数字之和。 Tips: 区间至少包括

    2024年02月11日
    浏览(30)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包