题意:每个人出场时获得等待时间*值的unhappy值。有个栈换出场顺序。问怎样最小?

一开始的时候觉得在中间取断点,dp[i][j]表示区间全出场后的最小值。那么dp[i][j]=dp[i][k]+dp[k+1][j],但这样是不行的。因为有可能最优解是[i][k]只出场部分,剩一些在栈里,然后再出场别的。

注意到栈有一个性质,如果第一个元素是第3个出栈,那么在它出栈前第2,3个元素已经出栈了。如果第一个元素第k出栈,那么[2,k]必在前k-1个出栈。那么状态转移时就列出所有可能的k。

正:

#pragma comment(linker,"/STACK:1024000000,1024000000")
#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<vector>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<set>
#include<algorithm>
#include <stack>
using namespace std;
const int SZ=1e2+,INF=0x7FFFFFFF;
typedef long long lon;
int n,arr[SZ],sum[SZ],dp[SZ][SZ]; void init()
{
cin>>n;
for(int i=;i<=n;++i)cin>>arr[i],sum[i]=sum[i-]+arr[i];
for(int len=;len<=n;++len)
{
for(int i=;i<=n;++i)
{
int r=i+len-;
if(r>n)break;
dp[i][r]=INF;
for(int k=;k<=len;++k)
{
dp[i][r]=min(dp[i][r],dp[i+][i+k-]+dp[i+k][r]+(k-)*arr[i]+(k)*(sum[r]-sum[i+k-]));
//if(i==1&&r==3)cout<<"dp[i][r]: "<<dp[i][r]<<" "<<dp[i+k][r]<<endl;
}
}
}
// for(int i=1;i<=n;++i)
// {
// for(int j=i;j<=n;++j)
// {
// cout<<"i: "<<i<<"j: "<<j<<" "<<dp[i][j]<<" ";
// }cout<<endl;
// }
cout<<dp[][n]<<endl;
} int main()
{
std::ios::sync_with_stdio();
int casenum;
cin>>casenum;
for(int time=;time<=casenum;++time)
{
cout<<"Case #"<<time<<": ";
init();
}
return ;
}

误:

#pragma comment(linker,"/STACK:1024000000,1024000000")
#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<vector>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<set>
#include<algorithm>
#include <stack>
using namespace std;
const int SZ=1e2+,INF=0x7FFFFFFF;
typedef long long lon;
int n,arr[SZ],sum[SZ],dp[SZ][SZ],st[SZ][SZ]; void init()
{
cin>>n;
for(int i=;i<=n;++i)cin>>arr[i],sum[i]=sum[i-]+arr[i];
for(int len=;len<=n;++len)
{
for(int i=;i<=n;++i)
{
int r=i+len-;
if(r>n)break;
st[i][r]=st[i+][r]+(len-)*arr[i];
}
}
for(int len=;len<=n;++len)
{
for(int i=;i<=n;++i)
{
int r=i+len-;
if(r>n)break;
dp[i][r]=st[i][r-]+(sum[r-]-sum[i-]);
if(i==&&r==)cout<<"h: "<<dp[i][r]<<endl;
for(int k=i;k<r;++k)
{
dp[i][r]=min(dp[i][r],dp[i][k]+st[k+][r-]+max(,(k-i+))*(sum[r-]-sum[k])+(k-i+)*arr[r]);
if(i==&&r==)cout<<"h: "<<dp[i][r]<<endl;
}
}
}
// for(int i=1;i<=n;++i)
// {
// for(int j=i;j<=n;++j)
// {
// cout<<"i: "<<i<<"j: "<<j<<" "<<dp[i][j]<<" ";
// }cout<<endl;
// }
cout<<dp[][n]<<endl;
} int main()
{
std::ios::sync_with_stdio();
int casenum;
cin>>casenum;
for(int time=;time<=casenum;++time)
{
cout<<"Case #"<<time<<": ";
init();
}
return ;
}

hdoj4283 You Are the One的更多相关文章

  1. 专题训练之区间DP

    例题:以下例题部分的内容来自https://blog.csdn.net/my_sunshine26/article/details/77141398 一.石子合并问题 1.(NYOJ737)http: ...

随机推荐

  1. Java设计模式应用——备忘录模式

    备忘录模式主要用于存档.游戏中我们打boss前总会存档,如果打boss失败,则读取存档,重新挑战boss. 可以看出来,备忘录模式一般包括如下数据结构 1. 存档文件:用于恢复备份场景的必要数据: 2 ...

  2. div居中布局

    利用margin属性可以实现div居中布局,把div的左边距和右边距设置为auto即可,代码如下 <!DOCTYPE html> <html> <head> < ...

  3. 计算概论(A)/基础编程练习1(8题)/4:求一元二次方程的根

    #include<stdio.h> #include<math.h> int main() { // 待解方程数目 int n; scanf("%d", & ...

  4. java网络基础知识的简述

    TCP/UDP的介绍 TCP协议:面向连接的,字节流无差错地传输协议. UDP协议:一个不可靠的无连接的数据传输协议. 说明:TCP可以想象成电话通讯,双方在通话时必须建立连接,一方没听清,会要求对方 ...

  5. Python入门之安装numpy和pandas

    最近要对一系列数据做同比比较,需要用到numpy和pandas来计算,不过使用python安装numpy和pandas因为linux环境没有外网遇到了很多问题就记下来了. 首要条件,python版本必 ...

  6. dba和运维专家们说有丰富的大型分布式系统架构设计经验纯属扯淡

    如果,一开始就从事dba和运维的专家们说他们有丰富的大型分布式系统架构设计经验,那纯属扯淡.除非,他们从是从开发专家或者架构师转型而来,那么他们才有资格说自己有丰富的大型分布式系统架构设计经验. 运维 ...

  7. 微信小程序编写新闻阅读列表

    微信小程序编写新闻阅读列表 不忘初心,方得始终:初心易得,始终难守. 本篇学习主要内容 Swiper 组件(轮播图) App.json 里的关于导航栏.标题的配置. Page 页面与应用程序的生命周期 ...

  8. 08: 查看Linux系统基本信息和硬盘CPU等

    目录: 1.1 查看Linux系统基本信息 1.2 查看三秒内的平均CPU 1.3 查看内存使用情况 1.4 查看当前系统负载 1.1 查看Linux系统基本信息返回顶部 1.查看Linux系统uui ...

  9. PHP安装包TS和NTS的区别

    原文链接:http://blog.csdn.net/zhuifengshenku/article/details/38796555 TS指Thread Safety,即线程安全,一般在IIS以ISAP ...

  10. 20165211 2017-2018-2 《Java程序设计》第3周学习总结

    20165211 2017-2018-2 <Java程序设计>第3周学习总结 教材学习内容总结 本周,我学习了书本上第四章的内容,以下是我整理的主要知识. 第四章 类与对象 编程语言的几个 ...