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

分析 : 

有大佬给出了四边形不等式优化........

发现并不会,于是开始学习区间DP的写法

对于某一个特定的区间 (i, j) 其合并成一个石子是从

某两个属于这个区间且连续不相交区间合并而来

即假设有断点 k 则 (i, j) = (i, k) + (k+1, j) + Sum(i, j)

这里定义 Sum(i, j) 为 i 到 j 这个区间石子的总和、前缀和可实现

只要枚举断点就可以知道 (i, j) 的最优值是多少了

定义 dp[i][j] = 区间 i 到 j 的石子合并代价的最值

转移方程就是枚举断点 dp[i][j] = dp[i][k] + dp[k+1][j] + Sum(i, j)

但是这个有一个坑,如果你用三重循环,分别表示

区间开头 i、区间结尾 j、区间断点 k 来进行 状态转移

这样是错误的,因为 dp[k+1][j] 在这样的循环下是还未确定的

所以有个技巧就是将一重循环变成区间长度,即

区间长度 len、区间开头 i、区间断点 k

还有一个问题就是,题目给出来的石头是链装的

只要“断环为链”即复制一份黏到末尾就行了

#include<bits/stdc++.h>
using namespace std;
const int INF = 0x3f3f3f3f;
;
int dp1[maxn][maxn], dp2[maxn][maxn];
int PreSum[maxn];
int arr[maxn];
int N;

int main(void)
{
    scanf("%d", &N);

    memset(dp1, , sizeof(dp1));
    memset(dp2, , sizeof(dp2));
    memset(PreSum, , sizeof(PreSum));

    ; i<=N; i++){
        scanf("%d", &arr[i]);
        arr[i+N] = arr[i];
    }

    ; i<=(N<<); i++)
        PreSum[i] = PreSum[i-] + arr[i];

    ; len<=N; len++){
        ; i<=(N<<)-len+; i++){
            ;
            , MM = INF;
            for(int k=i; k<j; k++){
                MM = min(MM, dp1[i][k]+dp1[k+][j]+PreSum[j]-PreSum[i-]);
                MX = max(MX, dp2[i][k]+dp2[k+][j]+PreSum[j]-PreSum[i-]);
            }
            dp1[i][j] = MM;
            dp2[i][j] = MX;
        }
    }

    , MM = INF;
    ; i<=N; i++){
        MM = min(MM, dp1[i][i+N-]);
        MX = max(MX, dp2[i][i+N-]);
    }

    printf("%d\n%d\n", MM, MX);
    ;
}

洛谷 P1080 石子合并 ( 区间DP )的更多相关文章

  1. 洛谷P1880 石子合并(区间DP)(环形DP)

    To 洛谷.1880 石子合并 题目描述 在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1 ...

  2. 洛谷 P1880 [NOI1995] 石子合并(区间DP)

    传送门 https://www.cnblogs.com/violet-acmer/p/9852294.html 题解: 这道题是石子合并问题稍微升级版 这道题和经典石子合并问题的不同在于,经典的石子合 ...

  3. 洛谷P1880 石子合并(环形石子合并 区间DP)

    题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1个算法,计算出将N堆石子合并成1 ...

  4. 经典DP 洛谷p1880 石子合并

    https://www.luogu.org/problemnew/show/P1880 题目 题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新 ...

  5. 石子合并 区间dp模板

    题意:中文题 Description 在操场上沿一直线排列着 n堆石子.现要将石子有次序地合并成一堆.规定每次只能选相邻的两堆石子合并成新的一堆, 并将新的一堆石子数记为该次合并的得分.允许在第一次合 ...

  6. HDU4632 Poj2955 括号匹配 整数划分 P1880 [NOI1995]石子合并 区间DP总结

    题意:给定一个字符串 输出回文子序列的个数    一个字符也算一个回文 很明显的区间dp  就是要往区间小的压缩! #include<bits/stdc++.h> using namesp ...

  7. 洛谷 P1880 石子合并

    题目描述 在一个圆形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1个算法,计算出将N堆石子合并成1 ...

  8. 洛谷P1040 加分二叉树(区间dp)

    P1040 加分二叉树 题目描述 设一个n个节点的二叉树tree的中序遍历为(1,2,3,…,n),其中数字1,2,3,…,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di, ...

  9. 石子合并 区间DP模板题

    题目链接:https://vjudge.net/problem/51Nod-1021 题意 N堆石子摆成一条线.现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石 ...

随机推荐

  1. numpy中线性代数用法

    numpy中线性代数用法 矩阵乘法 >>> import numpy as np >>> x=np.array([[1,2,3],[4,5,6]]) >> ...

  2. py3.7安装Scrapy及安装时的 Running setup.py install for Twisted ... error 和安装后的 Unhandled error in Deferred:

    1.首先,win+r 进入cmd,打开命令提示符,输入  pip install scrapy  等待自动安装: 2.到了后半段会出现  Running setup.py install for Tw ...

  3. springboot基于CORS处理跨域问题

    1. 为什么有跨域问题 跨域不一定都会有跨域问题. 因为跨域问题是浏览器对于ajax请求的一种安全限制:一个页面发起的ajax请求,只能是与当前页域名相同的路径,这能有效的阻止跨站攻击. 因此:跨域问 ...

  4. 實現QQ第三方登錄

    <?php // 写几个函数,分别用于获取code,token,openid,用户信息 // 跳转到QQ授权登录页面 function code(){ $response_type='code' ...

  5. RabbitMQ入门教程(十六):RabbitMQ与Spring集成

    原文:RabbitMQ入门教程(十六):RabbitMQ与Spring集成 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https: ...

  6. 解决Response.AddHeader中文乱码问题

    string filename = HttpUtility.UrlEncode(Encoding.UTF8.GetBytes("培训班自然情况表")); Response.AddH ...

  7. qt QAbstractItemModel一些方法介绍

    一. virtual bool setData(const QModelIndex &index, const QVariant &value, int role = Qt::Edit ...

  8. java复习(1)面向对象

    一.面向对象的概念 ----------------------------------------------------- 1.理解面向对象:(1)面向对象是相对于面向过程的语言 (2)面向对象和 ...

  9. vue.js(5)--事件修饰符

    vue中的事件修饰符(.stop..prevent..self..capture..once) (1)实例代码 <!DOCTYPE html> <html lang="en ...

  10. Linux文件读写笔记

    读文件: #include <stdio.h> #include <stdlib.h> #include <unistd.h> //linux下面的头文件 #inc ...