UESTC_摩天轮 2015 UESTC Training for Dynamic Programming<Problem K>
K - 摩天轮
Time Limit: 10000/4000MS (Java/Others) Memory Limit: 262143/262143KB (Java/Others)
一天,冬马被春希和雪菜拉着去一起去游乐园玩。
经过了各种过山车的洗礼后,三人决定去坐摩天轮休息下。
这是一个巨大的摩天轮,每一个车厢能坐任意多的人。现在,等着坐摩天轮的有n个人(包含他们3人),摩天轮还有m个车厢可以坐人。每个人都有自己肥胖程度,出于某些原因,胖子和瘦子坐在同一节车厢就会产生一定的矛盾,这个矛盾的值为(MAX−MIN)2,其中MAX为当前车厢里面最胖的人的肥胖程度,MIN为最廋的那个人的肥胖程度。
爱管闲事的春希当然不希望就因为这点小事而使大家的这趟旅途不愉快,于是他决定帮大家安排怎么坐才能使总的矛盾值最小,希望你能帮他找到这个最小的矛盾值
Input
第一行为两个整数n,m,分别表示人数和车厢数。(3≤n≤10000,1≤m≤5000)
第二行为n个整数,wi表示第i个人的肥胖程度。(0≤wi≤1000000)
Output
每组数据,输出一个整数,为矛盾的最小值。(答案保证小于231)
Sample input and output
| Sample Input | Sample Output |
|---|---|
4 2 |
18 |
解题思路:
这是一道斜率优化DP的题目.
我们令 f ( i , j ) 表示将前 i 个人分成 j 份所需的最小费用.
F ( i , j ) = min{ F ( u , j -1) + (sum[i] – sum[ u + 1])^2 }
但是这样做的复杂度高达O(N^2*M),对于本题的规模来讲无法承受,我们考虑
K1 < k2 < i ,且 k2 比 k1更优
有式子
F ( k2 , j - 1 ) + sum( k2 ) ^ 2 - F( k1 , j -1 ) – sum(k1+1) ^ 2
————————————————————— ————————————— < 2 * sum(i)
sum ( k2 ) – sum( k1+1)
显然这是一个斜率的式子.
假设现在 K(k1 – k2) < 2*sum[i] ,那么k1这个点还有价值吗?
我们注意到sum[i]是单调不减的,也就是说,对于之后的i2,i3…..
K(k1 – k2) < 2*sum[i_x] ,也就是说,k1 完全不可能成为某个点的最优解了,是可以舍弃的.
那么如果现在有 K(k1 – k2) >= 2*sum[i]呢,目前显然是选择k1比k2好,我们应不应该舍弃k2呢?答案是不应该,因为随着i的增大,sum[i]是不减的(也就是说,sum[i]可能会增大),此时 K(k1 – k2) < 2*sum[i] 是可能成立的.
综上所述,我们维护一个单调队列即可,只不过这个单调队列剔除 / 加入元素的条件都是和与斜率有关,这样我们可以在O(1)的时间内得到某个点的最优决策点,复杂度成功降到了O(n*m).
代码使用了滚动数组进行优化
#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
using namespace std;
const int maxn = 1e4 + ;
int n,m,h[maxn],f[][maxn],cur=,q[maxn]; double slope(int x,int y)
{
if (h[x+] == h[y+])
return 1e233;
return (double)(f[cur^][x] + h[x+]*h[x+] - (f[cur^][y] + h[y+]*h[y+])) / (double)(h[x+] - h[y+]);
} int main(int argc,char *argv[])
{
scanf("%d%d",&n,&m);
for(int i = ; i <= n ; ++ i) scanf("%d",&h[i]);sort(h+,h++n);
for(int i = ; i <= n ; ++ i) f[cur][i] = (h[i]-h[])*(h[i]-h[]);
for(int i = ; i <= m ; ++ i)
{
cur ^= ;
int front = , rear = ;
q[rear++] = ;
for(int j = ; j <= n ; ++ j)
{
while (rear - front > && slope(q[front],q[front+]) <= *h[j])
front++;
f[cur][j] = f[cur^][q[front]] + (h[j] - h[q[front]+])*(h[j] - h[q[front]+]);
while(rear - front > && slope(q[rear-],q[rear-]) >= slope(q[rear-],j))
rear--;
q[rear++] = j;
}
}
printf("%d\n",f[cur][n]);
return ;
}
UESTC_摩天轮 2015 UESTC Training for Dynamic Programming<Problem K>的更多相关文章
- UESTC_导弹拦截 2015 UESTC Training for Dynamic Programming<Problem N>
N - 导弹拦截 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submit ...
- UESTC_最少花费 2015 UESTC Training for Dynamic Programming<Problem D>
D - 最少花费 Time Limit: 30000/10000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submi ...
- UESTC_酱神寻宝 2015 UESTC Training for Dynamic Programming<Problem O>
O - 酱神寻宝 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submit ...
- UESTC_酱神的旅行 2015 UESTC Training for Dynamic Programming<Problem M>
M - 酱神的旅行 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submit ...
- UESTC_菲波拉契数制升级版 2015 UESTC Training for Dynamic Programming<Problem L>
L - 菲波拉契数制升级版 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Su ...
- UESTC_男神的约会 2015 UESTC Training for Dynamic Programming<Problem J>
J - 男神的约会 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submit ...
- UESTC_邱老师选妹子(二) 2015 UESTC Training for Dynamic Programming<Problem I>
I - 邱老师选妹子(二) Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Su ...
- UESTC_邱老师选妹子 2015 UESTC Training for Dynamic Programming<Problem H>
H - 邱老师选妹子 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submi ...
- UESTC_邱老师玩游戏 2015 UESTC Training for Dynamic Programming<Problem G>
G - 邱老师玩游戏 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submi ...
随机推荐
- 【转】各个层次的gcc警告 #pragma GCC diagnostic ignored "-Wunused-parameter" --不错
原文网址:http://blog.csdn.net/lizzywu/article/details/9419145 各个层次的gcc警告从上到下覆盖 变量(代码)级:指定某个变量警告 int a __ ...
- POJ 2392 Space Elevator DP
该题与POJ 1742的思路基本一致:http://www.cnblogs.com/sevenun/p/5442279.html(多重背包) 题意:给你n个电梯,第i个电梯高h[i],数量有c[i]个 ...
- 格而知之15:我所理解的Block(1)
1.Block 本质上是一个struct结构体,在这个结构体中,最重要的成员是一个函数(当然除函数外还有其他重要的成员). 2.在开始解析Block之前,首先来回顾一下Block的格式.Block相关 ...
- php字符串标点等字符截取不乱吗 封装方法
方法一: /** +---------------------------------------------------------- * 功能:字符串截取指定长度 * leo.li hen ...
- UISwitch + UIimage - 初识IOS
这里解释一个小例子,希望对你有点帮助,利用UISwitch控制UIimage的动画效果 先定义一个数组,用来存放照片,现在定义数组有一个特别简单的方法: NSArray *image1 = @[]; ...
- React-Native OpenGL体验二
搞了一下午,终于做了几个Demo出来,下面我就说一下我对React-Native下的OpenGL的流畅度的体验吧. 我使用的测试机是坚果手机 ...
- PHP学习笔记十六【方法】
<?php //给一个函数传递基本数据类型 $a=90; $b=90.8; $c=true; $d="hello world"; function test1($a,$b,$ ...
- VS 2003 无法打开Web项目 文件路径与URL不符 这两者需要映射到相同的服务器位置
解决方法: 将C:\Documents and Settings\Administrator\VSWebCache下面的文件全部删除
- js-String
1.一个字符串可以使用单引号或双引号 2.查找 字符串使用 indexOf() 来定位字符串中某一个指定的字符首次出现的位置 如果没找到对应的字符函数返回-1 lastIndexOf() 方法在字符串 ...
- 在[self addsubView:xxx]中,self.name 和 _name的区别
在[self addsubView:xxx]中,self.name 和 _name的区别self.name 会调用重写的getter方法,而_name添加的只是_name 这个成员变量