[BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化)
[BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化)
题面

分析
dp方程推导
显然,必然存在一种最优的买卖方案满足:每次买进操作使用完所有的人民币;每次卖出操作卖出所有的金券.
设dp[i]表示第i天卖出所有金券的能够得到的钱数。则有:
\]
意义是第j天按R[j]的比例用所有钱买入股票,然后在第i天全部卖出,比例仍为R[j]
去掉max,把i,j分开移项,得
\]
令\(x[j]=\frac{dp[j]}{A[j] \times R[j]+B[j] }\),\(y[j]=\frac{dp[j] \times R[j]}{A[j] \times R[j]+B[j] }\)
那么状态转移方程就变成了\(dp[i]=x[j] \times B[i]+y[j] \times A[i]\)
变换成直线方程的形式,y[j]前面有个系数A[i],把它除掉,得
\]
因此用斜率为\(-\frac{B[i]}{A[i]}\)的直线经过点\((x[j],y[j])\),解出的最小截距就是\(dp[i]\)的值。注意到\(-\frac{B[i]}{A[i]}\)和x[j]并不单调,不能用单调队列优化。考虑cdq分治求解
cdq 分治
cdq分治求解斜率优化其实类似一个三维偏序过程,第一维是点i对应的直线斜率(本题中是\(-\frac{B[i]}{A[i]}\)),第二,三维是对应的点的坐标
首先在cdq分治的时候需要维护一个数组a,记录a对应的dp值下标id,和它对应点的坐标。先把a按直线斜率排序,然后开始cdq分治。
当我们分治到区间[l,r]的时候,我们执行以下伪代码
procedure CDQ(l,r){
if(l==r){
用a[l]对应的dp值来更新a的(x,y)
return
}
mid=(l+r)/2
把[l,r]内的元素按id分成两个区间,id<=mid的分到[l,mid],否则分到[mid+1,r]
CDQ(l,mid),先递归计算id值小的
把[l,mid]区间对应的点建出凸壳,用于下一步更新[mid+1,r]的dp值
for(k in [mid+1,r]){
用单调队列更新a[k]对应的dp值,因为我们cdq之前按斜率排过序,所以不会出问题
注意到在这里面即使a[k]对应的dp值的最优决策不在[mid+1,r]中,那么回溯到一个更大的区间后,也会被更新到
}
CDQ(mid+1,r),继续递归下去更新
按x,y归并排序[l,mid],[mid+1,r],这样回溯的时候x单调递增,才可以直接单调栈(队列)建凸壳
}
注意精度问题,以及斜率为\(+ ∞\)的情况
代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define eps 1e-8
#define INF 1e18
#define maxn 100000
using namespace std;
typedef long long ll;
typedef double db;
int n;
db S;
db A[maxn+5],B[maxn+5],R[maxn+5];
struct Vector{
int id;
db x;
db y;
Vector(){
}
Vector(int _id,db _x,db _y){
id=_id;
x=_x;
y=_y;
}
}a[maxn+5];
typedef Vector point;
db slope(point p,point q){
db down=p.x-q.x;
db up=p.y-q.y;
if(fabsl(down)<=eps) return INF;
else return up/down;
}
int cmp1(point p,point q){
db k1=(fabs(A[p.id])>eps)?-B[p.id]/A[p.id]:INF;
db k2=(fabs(A[q.id])>eps)?-B[q.id]/A[q.id]:INF;
return k1<k2;//按斜率排序,保证斜率单调
}
int cmp2(point p,point q){//按x,y归并排序
if(p.x==q.x) return p.y<q.y;
else return p.x<q.x;
}
int head,tail;
point q[maxn+5];
point tmp[maxn+5];
db dp[maxn+5];
db get_x(int j){
return dp[j]/(A[j]*R[j]+B[j]);
}
db get_y(int j){
return dp[j]*R[j]/(A[j]*R[j]+B[j]);
}
void cdq_divide(int l,int r){
if(l==r){
dp[a[l].id]=max(dp[a[l].id],dp[a[l].id-1]);
a[l].x=get_x(a[l].id);
a[l].y=get_y(a[l].id);
return;
}
int mid=(l+r)>>1;
int pl=l,pr=mid+1;
for(int i=l;i<=r;i++){
if(a[i].id<=mid) tmp[pl++]=a[i];
else tmp[pr++]=a[i];
}
for(int i=l;i<=r;i++) a[i]=tmp[i];
cdq_divide(l,mid);
head=1,tail=0;
for(int i=l;i<=mid;i++){
while(head<tail&&slope(q[tail-1],q[tail])<=slope(q[tail-1],a[i])) tail--;
q[++tail]=a[i];
}
for(int k=r;k>mid;k--){//由于是维护上凸壳,斜率从小到大排,要倒序更新
int i=a[k].id;
db s2=(fabs(A[i])>eps)?-B[i]/A[i]:INF;
while(head<tail&&slope(q[head],q[head+1])>=s2) head++;
int j=q[head].id;
dp[i]=max(dp[i],A[i]*get_y(j)+B[i]*get_x(j));
// dp[i]=max(dp[i],dp[i-1]);
}
cdq_divide(mid+1,r);
int num=l-1;
pl=l,pr=mid+1;
while(pl<=mid&&pr<=r){//按x,y归并排序
if(cmp2(a[pl],a[pr])) tmp[++num]=a[pl++];
else tmp[++num]=a[pr++];
}
while(pl<=mid) tmp[++num]=a[pl++];
while(pr<=r) tmp[++num]=a[pr++];
for(int i=l;i<=r;i++) a[i]=tmp[i];
}
int main(){
// freopen("5.in","r",stdin);
scanf("%d %lf",&n,&S);
for(int i=1;i<=n;i++){
scanf("%lf %lf %lf",&A[i],&B[i],&R[i]);
}
dp[0]=S;
for(int i=0;i<=n;i++){
a[i].id=i;
}
sort(a,a+1+n,cmp1);
cdq_divide(0,n);
printf("%.3lf\n",dp[n]);
}
[BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化)的更多相关文章
- bzoj1492[NOI2007]货币兑换Cash cdq分治+斜率优化dp
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 5541 Solved: 2228[Submit][Sta ...
- BZOJ 1492: [NOI2007]货币兑换Cash [CDQ分治 斜率优化DP]
传送门 题意:不想写... 扔链接就跑 好吧我回来了 首先发现每次兑换一定是全部兑换,因为你兑换说明有利可图,是为了后面的某一天两种卷的汇率差别明显而兑换 那么一定拿全利啊,一定比多天的组合好 $f[ ...
- BZOJ1492:[NOI2007]货币兑换 (CDQ分治+斜率优化DP | splay动态维护凸包)
BZOJ1492:[NOI2007]货币兑换 题目传送门 [问题描述] 小Y最近在一家金券交易所工作.该金券交易所只发行交易两种金券:A纪念券(以下简称A券)和B纪念券(以下简称B券).每个持有金券的 ...
- BZOJ1492: [NOI2007]货币兑换Cash(CDQ分治,斜率优化动态规划)
Description 小Y最近在一家金券交易所工作.该金券交易所只发行交易两种金券:A纪念券(以下简称A券)和 B纪念券(以下 简称B券).每个持有金券的顾客都有一个自己的帐户.金券的数目可以是一个 ...
- bzoj 1492: [NOI2007]货币兑换Cash【贪心+斜率优化dp+cdq】
参考:http://www.cnblogs.com/lidaxin/p/5240220.html 虽然splay会方便很多,但是懒得写,于是写了cdq 首先要想到贪心的思路,因为如果在某天买入是能得到 ...
- BZOJ 1492 [NOI2007]货币兑换Cash (CDQ分治/splay 维护凸包)
题目大意:太长了略 splay调了两天一直WA弃疗了 首先,我们可以猜一个贪心,如果买/卖,就一定都买/卖掉,否则不买/卖 反正货币的行情都是已知的,没有任何风险,所以肯定要选择最最最优的方案了 容易 ...
- 【uoj#244】[UER #7]短路 CDQ分治+斜率优化dp
题目描述 给出 $(2n+1)\times (2n+1)$ 个点,点 $(i,j)$ 的权值为 $a[max(|i-n-1|,|j-n-1|)]$ ,找一条从 $(1,1)$ 走到 $(2n+1,2n ...
- [BZOJ1492][NOI2007]货币兑换Cash(斜率优化+CDQ分治)
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 5838 Solved: 2345[Submit][Sta ...
- [BZOJ1492] [NOI2007]货币兑换Cash 斜率优化+cdq/平衡树维护凸包
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 5907 Solved: 2377[Submit][Sta ...
随机推荐
- git log 详解 以及代码量统计
https://git-scm.com/book/zh/v1/Git-%E5%9F%BA%E7%A1%80-%E6%9F%A5%E7%9C%8B%E6%8F%90%E4%BA%A4%E5%8E%86% ...
- hadoop中yarn
一.yarn的概述 Apache Yarn(Yet Another Resource Negotiator的缩写)是hadoop集群资源管理器系统,Yarn从hadoop 2引入,最初是为了改善Map ...
- node.js 的 中间件 初理解
听说中间件还挺重要,下面梳理一下初认识: 中间件是什么?简单说说http请求服务的过滤,当交给函数处理之前先交给它处理.匹配后会终止,要想再匹配,得加: next. 中间件能解决什么问题?检测用户登录 ...
- C++11 lambda表达式小结
目录 简介 结构 return type parameter list capture list 值捕获和引用捕获 变量修改 隐式和显式捕获 捕获列表小结: problems 1.为什么需要使用mut ...
- 转载--C 的回归
转载自http://blog.codingnow.com/2007/09/c_vs_cplusplus.html 周末出差,去另一个城市给公司的一个项目解决点问题.回程去机场的路上,我用手机上 goo ...
- 安装JDK ,提示 错误1316 指定的账户已存在
基于情况: 安装了一个JDK 后,在文件目录中删除了相关文件,之后再次安装,提示 错误1316 指定的账户已存在 造成原因:安装JDK,相当于安装了一个软件,要使用系统的软件卸载功能卸载,不能只删除 ...
- 嵌入式Linux之虚拟内存地址空间布局(Virtual Memory Space)
虚拟内存地址空间 Linux内核属于微内核的范畴,内核控制计算机的硬件资源,运行在特权模式:用户态应用程序运行在普通用户模式,无法直接访问硬件资源,必须依托于内核提供的资源,如CPU资源.Memory ...
- yield(放弃、谦逊、礼让) - 瞬时的,暂时放了马上再抢
两个线程抢占CPU各自执行任务,代码如下: public class Demo03 { public static void main(String[] args) throws Interrupte ...
- docker 提高效率 network-bridging 桥接
安装的时间顺序 bit3 192.168.107.128 wredis 192.168.107.129 wmysql 192.168.107.130 wslave 192.168.107.131 w ...
- PHP执行外部程序
备份/恢复数据库 exec - 执行一个外部程序(在php文件所在目录进行执行) 很久以前写的,很多方法是项目中的直接复制粘体用不了,只能提供下思路. 用到执行外部程序的就这一句: exec(&quo ...