bzoj4247挂饰——压缩的动态规划

时间:2022-12-30 16:42:24

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4247

1.dp之前要先按挂钩个数从大到小排序,不然挂钩一度用成负的也可能是正确的,不仅脚标难存,而且不知道各种时刻负多少以内是合法的。

2.但是最多有4000000个挂钩,省掉一维勉强开下数组也就罢了,会TLE!

  考虑多余n=2000个的挂钩就不会被用上了,所以第二维开成2000;

  (——状态表示至少有 j 个挂钩!!!)

  那么当a [ i ] > j 的时候怎么办呢?大家都是这么写的:

  d [ i ] [ j ] = max( d [ i -1 ] [ j ] , d [ i - 1 ] [ max( j - a [ i ] , 0) +1 ] + b [ i ] );

  也就是若a [ i ] > j,则d [ i ] [ j ] = max( d [ i -1 ] [ j ] , d [ i - 1 ] [ 1 ] + b [ i ] ),然后把多出来的挂钩忽略;

  为什么一定是1而不是k(1<=k<=j)呢?因为d [ ] [ 1 ]一定是最大的值!

  这是因为d [ ] [ 1 ]其实可能有很多挂钩,但把多余挂钩都忽略了,所以记录的其实是所有剩余挂钩中的最大值;

  那么0也是啊?

    但0的话不能再挂东西了,而只要剩下1个挂钩,就可以往上挂下一个东西,所以1可以一直被更新;

  如果剩下的东西都是0挂钩怎么办?

    所以才不能只记录d [ ] [ 1 ],而要记录到n,这样该挂饰实际有多个挂钩的优势也不会被忽略了!

3.所以到最后1表示有剩余挂钩的最大值,0表示无剩余挂钩的最大值;

  但其实每次都会用1的值更新0的值,所以最后直接输出0的值就行了!也就是像状态定义的一样,0包含了所有情况。

4.突然想到,那么即使当 j - a [ i ] > 0,也不一定用 d [ i -1 ] [ j - a [ i ] ]来更新,只需用 k ( j - a [ i ] <= k <= j )然后忽略多余挂钩就行了;

  那为什么不这样呢?因为 j 的值越大,相当于对挂钩数的限制越严,卡掉的状态也越多,值就越来越不优了!

  所以尽量用靠前的 j 值来更新!

5.所以应该是n^2的复杂度才对,为什么是约6000ms?而且那种省掉一维的写法为什么不行?

updt(2018.9.18):当j==0时max(j-r[i].a,0)+1==1,会用到本层的值,应该用上一层的值。

        折中的方法是滚动数组。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int n;
long long d[],ans;
struct Node{
int a,c;
}r[];
bool cmp(Node x,Node y){return x.a>y.a;}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
scanf("%d%d",&r[i].a,&r[i].c);
sort(r+,r+n+,cmp);
memset(d,-,sizeof d);
d[]=;
for(int i=;i<=n;i++)
{
// printf("i=%d a[i]=%d c[i]=%d\n",i,r[i].a,r[i].c);
// lm+=r[i].a;
if(!r[i].a)//
for(int j=;j<=n;j++)
d[j]=max(d[j],d[j+]+r[i].c);
else for(int j=n;j>=;j--)
d[j]=max(d[j],d[max(j-r[i].a,)+]+r[i].c);
}
ans=-2e9-;
for(int i=;i<=n;i++)//i=0
ans=max(ans,d[i]);
printf("%lld",ans);
return ;
}

省掉一维的WA

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int n;
long long d[][],ans;
struct Node{
int a,c;
}r[];
bool cmp(Node x,Node y){return x.a>y.a;}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
scanf("%d%d",&r[i].a,&r[i].c);
sort(r+,r+n+,cmp);
memset(d,-,sizeof d);
d[][]=;
for(int i=;i<=n;i++)
{
for(int j=;j<=n;j++)
d[i][j]=max(d[i-][j],d[i-][max(j-r[i].a,)+]+r[i].c);
}
// ans=-2e9-7;
// for(int i=0;i<=n;i++)//i=0
// ans=max(ans,d[n][i]);
// ans=max(d[n][0],d[n][1]);
printf("%lld",d[n][]);
return ;
}

bzoj4247挂饰——压缩的动态规划的更多相关文章

  1. &lbrack;BZOJ4247&rsqb;挂饰&lpar;DP&rpar;

    当最终挂饰集合确定了,一定是先挂挂钩多的在挂挂钩少的. 于是按挂钩从大到小排序,然后就是简单的01背包. #include<cstdio> #include<algorithm&gt ...

  2. BZOJ4247挂饰

    Description     JOI君有N个装在手机上的挂饰,编号为1...N. JOI君可以将其中的一些装在手机上.     JOI君的挂饰有一些与众不同--其中的一些挂饰附有可以挂其他挂件的挂钩 ...

  3. bzoj千题计划197:bzoj4247&colon; 挂饰

    http://www.lydsy.com/JudgeOnline/problem.php?id=4247 先把挂饰按挂钩数量从大到小排序 dp[i][j]前i个挂饰,剩下j个挂钩的最大喜悦值 分挂和不 ...

  4. BZOJ4247 &colon; 挂饰

    首先将挂饰按照挂钩个数从大到小排序,然后DP 设f[i][j]处理完前i个挂饰,还有j个多余挂钩的最大喜悦值,则 f[0][1]=0 f[i][j]=max(f[i-1][max(j-a[i],0)+ ...

  5. &lbrack;bzoj4247&rsqb;&lbrack;挂饰&rsqb; &lpar;动规&plus;排序&rpar;

    Description JOI君有N个装在手机上的挂饰,编号为1...N. JOI君可以将其中的一些装在手机上. JOI君的挂饰有一些与众不同——其中的一些挂饰附有可以挂其他挂件的挂钩.每个挂件要么直 ...

  6. bzoj4247&colon; 挂饰&lpar;背包dp&rpar;

    4247: 挂饰 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1136  Solved: 454[Submit][Status][Discuss] ...

  7. bzoj4247&colon; 挂饰&lpar;背包&rpar;

    4247: 挂饰 题目:传送门 题解: 看完题目很明显的一道二维背包(一开始还推错了) 设f[i][j]表示前i个挂饰选完(可以有不选)之后还剩下j个挂钩的最大值(j最多贡献为n) 那么f[i][j] ...

  8. BZOJ4247 挂饰(动态规划)

    相当于一个有负体积的背包.显然如果确定了选哪些,应该先把体积小的挂上去.于是按体积从小到大排序,就是一个裸的背包了. #include<iostream> #include<cstd ...

  9. bzoj4247挂饰——DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4247 就是01背包: 把挂钩数限制在n以内,因为不需要更多,而这会带来一些问题,就是有很多挂 ...

随机推荐

  1. elasticsearch【更新】操作

    基于上一篇博文基础上,进行es的操作,document的新增比较简单,就不说了,这里主要说说更新操作. 更新操作,有两大类,一个是Replace,一个是Update,就是说一个是替换,一个是更新. 替 ...

  2. Android开发规范——命名 &lpar;转)

    转自: http://blog.sina.com.cn/s/blog_3f5dd7810101j4u2.html 在讲解命名规范前,先初略介绍下当前主要的标识符命名法和英文缩写规则. 标识符命名法 标 ...

  3. iOS开发入门教程

    iOS开发入门教程 http://my.oschina.net/mailzwj/blog/133273 摘要 iOS开发入门教程,从创建项目到运行项目,包括OC基础,调试,模拟器设置等相关知识. iO ...

  4. POI导入excel时读取excel数据的真实行数

    有很多时候会出现空的数据导致行数被识别多的情况 // 获取Excel表的真实行数 int getExcelRealRow(Sheet sheet) { boolean flag = false; fo ...

  5. 统计C&sol;C&plus;&plus;代码行数

    近日在写一个统计项目中C/C++文件(后缀名:C/CPP/CC/H/HPP文件)代码行数的小程序.给定包含C/C++代码的目录,统计目录里所有C/C++文件的总代码行数.有效代码行数.注释行数.空白行 ...

  6. 理解ES7中的async&sol;await

    理解ES7中的async/await 优势是:就是解决多层异步回调的嵌套 从字面上理解 async/await, async是 "异步"的含义,await可以认为是 async w ...

  7. luoguU38228 签到题 &lpar;BSGS&rpar;

    签到一脸 $a_n=10a_{n-1}+1$求出通项$a_n=\frac{10^n-1}{9}$,然后可以化成$10^n=9K+1 (mod m)$,求一个最小的n 然后我们知道这个n一定是<= ...

  8. 企业内部在centos7&period;2系统中必杀技NTP时间服务器及内网服务器时间同步&lpar;windows和linux客户端同步&rpar;

    网络时间协议NTP(Network Time Protocol)是用于互联网中时间同步的标准互联网协议.NTP的用途是把计算机的时间同步到某些时间标准.目前采用的时间标准是世界协调时UTC(Unive ...

  9. jquery的radio和checkbox的标签的操作集合

    jquery的radio和checkbox的标签的操作集合: $("input[name='radio_name'][checked]").val(); //选择被选中Radio的 ...

  10. 国内首款 FPGA 云服务器,性能是通用 CPU 服务器 30 倍以上

    版权声明:本文由薛梁原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/628340001485134638 来源:腾云阁 ht ...