BZOJ1492 货币兑换 CDQ分治优化DP
1492: [NOI2007]货币兑换Cash
Time Limit: 5 Sec Memory Limit: 64 MB
Description
.png)
.png)
Input
对于40%的测试数据,满足N ≤10;
对于60%的测试数据,满足N ≤1 000;
对于100%的测试数据,满足N ≤100 000;
Output
只有一个实数MaxProfit,表示第N天的操作结束时能够获得的最大的金钱数目。答案保留3位小数。
Sample Input
1 1 1
1 2 2
2 2 3
Sample Output
HINT
.png)
(转载请注明原文地址:http://www.cnblogs.com/LadyLex/p/8028556.html )
终于把CDQ维护凸壳学了……
那么既然题目已经给提示2了,我们就可以针对性的设$f[i]$为第i天获得的最多A券数,$ans[i]$为第i天的最大获利,那么我们要求的就是ans[n]了。
这个dp显然是可以$O(n^{2})$解决的……但是这样拿不了后面的分数。
然后你就想啊,肯定要优化啊……然后你就可以化出一个斜率的式子来。
对于$ans[i]$的决策,我们设两天j和k,j比k优秀的话,就会有
$(f[j]-f[k])*a[i]+(f[j]/rate[j]-f[k]/rate[k])*b[i]>0$
再设$g[i]=f[i]/rate[i]$(也就是第i天获得的最多B券数),为了处理不等式的符号我们设$f[j]<f[k]$
所以有:
$(g[j]-g[k])*b[i]>-(f[j]-f[k])*a[i]$
$(g[j]-g[k])/(f[j]-f[k])<-a[i]/b[i]$
(当然,实际情况是有$f[j]==f[k]$,即斜率不存在的情况存在的,到时候还要讨论。)
那么我们转化到一些坐标为(f[i],g[i])的二维平面的点上来。
我们建立一个这些点的上凸壳,然后在凸壳上二分最靠左的最后一个满足$k(point(x),point(x+1))<-a[i]/b[i]$的点x,
那么$x+1$就是最优秀的取值,也即本次决策点。
但是你发现这个f[i]不随i单调……那么我们考虑splay或者CDQ
打个J的splay啊
那么我们CDQ维护凸壳并且决策就好了……
具体实现是让$f[i]$有序之后按照正常方法建凸壳,然后让$-a[i]/b[i]$有序(我是从大到小排序),单调扫一边完成决策。
怎么让这俩有序呢……一是大力sort,复杂度$O(nlog^{2}n)$,一是归并排序,复杂度$O(nlogn)$
两种方法我都打了一下……感觉少个$logn$没快到哪去,也没长到哪去……
如果复杂度可行还是打$log^{2}$吧……
两份代码:
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define eps 1e-8
#define N 100010
#define db double
#define inf 0x7fffffff
#define sign(a) (((a)>-eps)-((a)<eps))
int n,top,sta[N],id[N],tmp[N],idk[N],tmpk[N],match[N];
db ans[N],ak[N],bk[N],rate[N],f[N],g[N];
inline db max(db a,db b){return a>b?a:b;}
inline bool comp(const int &a,const int &b)
{return f[a]<f[b] || ( sign(f[a]-f[b])==&&g[a]<g[b] );}
inline double k(int a,int b)
{
if(sign(f[a]-f[b])==)return sign(g[a]-g[b])*inf;
return (g[a]-g[b])/(f[a]-f[b]);
}
inline bool compk(const int &a,const int &b)
{return sign( (-ak[a]/bk[a]) - (-ak[b]/bk[b]) ) > ;}
inline void CDQ(int l,int r)
{
if(l==r){g[l]=f[l]/rate[l];return;}
register int hd1,i,t,mi=l+r>>,p=l-,q=mi,h=l-;
for(i=l;i<=r;++i)
if(match[idk[i]]<=mi)tmpk[++p]=idk[i];
else tmpk[++q]=idk[i];
for(i=l;i<=r;++i)idk[i]=tmpk[i];
CDQ(l,mi);
for(top=,i=l;i<=mi;++i)
{
while(top> && k(sta[top-],sta[top]) < k(sta[top],id[i]) )--top;
sta[++top]=id[i];
}
for(hd1=,i=mi+;i<=r;++i)
{
t=match[idk[i]];
while( hd1<top&&sign( k(sta[hd1],sta[hd1+]) - (-ak[t]/bk[t]) ) >= )++hd1;
ans[t]=max(ans[t],f[sta[hd1]]*ak[t]+g[sta[hd1]]*bk[t]);
}
for(i=mi+;i<=r;++i)
t=match[idk[i]],ans[t]=max(ans[t],ans[t-]),f[t]=ans[t]*rate[t]/(ak[t]*rate[t]+bk[t]);
CDQ(mi+,r);
p=l,q=mi+,h=l;
while(p<=mi&&q<=r)
if(comp(id[p],id[q]))tmp[h++]=id[p++];
else tmp[h++]=id[q++];
while(p<=mi)tmp[h++]=id[p++];
while(q<=r)tmp[h++]=id[q++];
for(i=l;i<=r;++i)id[i]=tmp[i];
}
int main()
{
register int i,j;
scanf("%d%lf",&n,&ans[]);
for(i=;i<=n;++i)
scanf("%lf%lf%lf",&ak[i],&bk[i],&rate[i]);
f[]=ans[]*rate[]/(ak[]*rate[]+bk[]);
for(i=;i<=n;++i)id[i]=idk[i]=match[i]=i;
sort(match+,match+n+,compk);
CDQ(,n);
printf("%.3f\n",ans[n]);
}
nlogn
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define eps 1e-8
#define N 100010
#define db double
#define inf 0x7fffffff
#define sign(a) (((a)>-eps)-((a)<eps))
int n;
db ans[N],ak[N],bk[N],rate[N],f[N],g[N];
inline db max(db a,db b){return a>b?a:b;}
int top,sta[N];
int id[N],tmp[N],idk[N];
inline bool comp(const int &a,const int &b)
{
return f[a]<f[b] || ( sign(f[a]-f[b])==&&g[a]<g[b] );
}
inline double k(int a,int b)
{
if(sign(f[a]-f[b])==)
return sign(g[a]-g[b])*inf;
return (g[a]-g[b])/(f[a]-f[b]);
}
inline bool compk(const int &a,const int &b)
{
return sign( (-ak[a]/bk[a]) - (-ak[b]/bk[b]) ) > ;
}
inline void CDQ(int l,int r)
{
if(l==r){g[l]=f[l]/rate[l];return;}
register int hd1,i,mi=l+r>>;
CDQ(l,mi);
for(i=mi+;i<=r;++i)id[i]=i;sort(id+l,id+mi+,comp);
for(i=mi+;i<=r;++i)idk[i]=i;sort(idk+mi+,idk+r+,compk);
for(top=,i=l;i<=mi;++i)
{
while(top> && k(sta[top-],sta[top]) < k(sta[top],id[i]) )--top;//****
sta[++top]=id[i];
}
for(hd1=,i=mi+;i<=r;++i)
{
while( hd1<top&&sign( k(sta[hd1],sta[hd1+]) - (-ak[idk[i]]/bk[idk[i]]) ) >= )++hd1;
ans[idk[i]]=max(ans[idk[i]],f[sta[hd1]]*ak[idk[i]]+g[sta[hd1]]*bk[idk[i]]);
}
for(i=mi+;i<=r;++i)
ans[i]=max(ans[i],ans[i-]),f[i]=ans[i]*rate[i]/(ak[i]*rate[i]+bk[i]);
CDQ(mi+,r);
}
int main()
{
register int i,j;
scanf("%d%lf",&n,&ans[]);
for(i=;i<=n;++i)
scanf("%lf%lf%lf",&ak[i],&bk[i],&rate[i]);
f[]=ans[]*rate[]/(ak[]*rate[]+bk[]);
for(i=;i<=n;++i)id[i]=i;
CDQ(,n);
printf("%.3f\n",ans[n]);
}
nlog^2n
BZOJ1492 货币兑换 CDQ分治优化DP的更多相关文章
- 洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP
洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP 题目描述 佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他. 玩具上有一个数列,数列中某些项的值可能会 ...
- SPOJ LIS2 - Another Longest Increasing Subsequence Problem(CDQ分治优化DP)
题目链接 LIS2 经典的三维偏序问题. 考虑$cdq$分治. 不过这题的顺序应该是 $cdq(l, mid)$ $solve(l, r)$ $cdq(mid+1, r)$ 因为有个$DP$. #i ...
- luogu4093 序列 (cdq分治优化dp)
设f[i]是以i位置为结尾的最长满足条件子序列的长度 那么j能转移到i的条件是,$j<i , max[j]<=a[i] , a[j]<=min[i]$,其中max和min表示这个位置 ...
- P4027 [NOI2007]货币兑换(斜率优化dp+cdq分治)
P4027 [NOI2007]货币兑换 显然,如果某一天要买券,一定是把钱全部花掉.否则不是最优(攒着干啥) 我们设$f[j]$为第$j$天时用户手上最多有多少钱 设$w$为花完钱买到的$B$券数 $ ...
- [NOI2007]货币兑换 cdq分治,斜率优化
[NOI2007]货币兑换 LG传送门 妥妥的\(n \log n\)cdq做法. 这题用cdq分治也可以\(n \log n\)但是在洛谷上竟然比一些优秀的splay跑得慢真是见了鬼了看来还是人丑常 ...
- 【BZOJ1492】【Luogu P4027】 [NOI2007]货币兑换 CDQ分治,平衡树,动态凸包
斜率在转移顺序下不满足单调性的斜率优化\(DP\),用动态凸包来维护.送命题. 简化版题意:每次在凸包上插入一个点,以及求一条斜率为\(K\)的直线与当前凸包的交点.思路简单实现困难. \(P.s\) ...
- BZOJ3963: [WF2011]MachineWorks 【CDQ+斜率优化DP】*
BZOJ3963: [WF2011]MachineWorks Description 你是任意性复杂机器公司(Arbitrarily Complex Machines, ACM)的经理,公司使用更加先 ...
- 【BZOJ 1492】 [NOI2007]货币兑换Cash 斜率优化DP
先说一下斜率优化:这是一种经典的dp优化,是OI中利用数形结合的思想解决问题的典范,通常用于优化dp,有时候其他的一些决策优化也会用到,看待他的角度一般有两种,但均将决策看为二维坐标系上的点,并转化为 ...
- SRM12 T2夏令营(分治优化DP+主席树 (已更新NKlogN)/ 线段树优化DP)
先写出朴素的DP方程f[i][j]=f[k][j-1]+h[k+1][i] {k<i}(h表示[k+1,j]有几个不同的数) 显然时间空间复杂度都无法承受 仔细想想可以发现对于一个点 i ...
随机推荐
- P1341 无序字母对
题目描述 给定n个各不相同的无序字母对(区分大小写,无序即字母对中的两个字母可以位置颠倒).请构造一个有n+1个字母的字符串使得每个字母对都在这个字符串中出现. 输入输出格式 输入格式: 第一行输入一 ...
- MVC 在action拦截器中获取当前进入的控制器和aciton名
我们在实现了action拦截器以后(继承至System.Web.Mvc.IActionFilter),需要在重写的方法OnActionExecuting中去获得当前进入的控制器和action名称,如何 ...
- 代码段:js表单提交检测
市面上当然有很多成型的框架,比如jquery的validation插件各种吧.现在工作地,由于前端童鞋也没用这些个插件.根据业务的要求,自己就在代码里写了个简单的表单提交的检测代码(php的也写了一个 ...
- Transaction Check Error:file /usr/libexec/getconf/default conflicts between attempted installs of gcc-6.4.1-1.fc25.i686 and gcc-6.4.1-1.fc25.x86_64
今天在我的ubuntu系统上使用yum来安装软件时出错了错误:Transaction Check Error:file /usr/libexec/getconf/default conflicts b ...
- 2017-2018-2 20155224『网络对抗技术』Exp8:Web基础
实践具体要求 Web前端HTML(0.5分) 能正常安装.启停Apache.理解HTML,理解表单,理解GET与POST方法,编写一个含有表单的HTML. Web前端javascipt(0.5分) 理 ...
- 【Win32 API】利用SendMessage实现winform与wpf之间的消息传递
原文:[Win32 API]利用SendMessage实现winform与wpf之间的消息传递 引言 有一次心血来潮,突然想研究一下进程间的通信,能够实现消息传递的方法有几种,其中win32ap ...
- 洛咕3312 [SDOI2014]数表
洛咕3312 [SDOI2014]数表 终于独立写出一道题了...真tm开心(还是先写完题解在写的) 先无视a的限制,设\(f[i]\)表示i的约数之和 不妨设\(n<m\) \(Ans=\su ...
- python list的一个面试题
面试题''' 一个list,里面的数字偶数在左边,奇数在右边,不借助其他列表 ''' def userlist(add_list): if type(add_list)==list: if len(a ...
- PWM输出
PWM(Pulse Width Modulation),脉冲宽度调制. 脉冲的频率由ARR控制,ARR越大频率越小:占空比由CCRx控制,CCRx越小占空比越大. 捕获/比较通道的输出部分(通道1) ...
- 异步编程之asyncio简单介绍
引言: python由于GIL(全局锁)的存在,不能发挥多核的优势,其性能一直饱受诟病.然而在IO密集型的网络编程里,异步处理比同步处理能提升成百上千倍的效率,弥补了python性能方面的短板. as ...