bzoj千题计划237:bzoj1492: [NOI2007]货币兑换Cash
http://www.lydsy.com/JudgeOnline/problem.php?id=1492
dp[i] 表示 第i天卖完的最大收益
朴素的dp:
枚举从哪一天买来的在第i天卖掉,或者是不操作
dp[i]=max(dp[i-1],X[j]*A[i]+Y[j]*B[i])
其中X[j]表示在第j天能买多少A纪念券,Y[j]表示在第j天能买多少B纪念券
可列方程 X[j]*A[j]+Y[j]*B[j]=dp[j]
又因为 X[j]=Rate[j]*Y[j]
所以解出 Y[j]=dp[j]/(B[j]+A[j]*Rate[j])
优化:
dp[i]=X[j]*A[i]+Y[j]*B[i]
Y[j]=dp[i]/B[i] - A[i]/B[i] * x[j]
斜率优化形式,维护上凸壳,最大化截距
点:(X[j],Y[j])
斜率:-A[i]/B[i]
但是 -A[i]/B[i] 不具有单调性
所以不能用单调队列维护斜率
可以用平衡树维护,O(n * logn * logn)
更简单的方法:CDQ分治
假设现在正在计算 dp[l]~dp[r],即solve(l,r)
对于每个j∈[l,r],[1,j-1]都是j的一种决策
令 mid=(l+r)/ 2
我们先计算出 dp[l]~dp[mid],然后用这些去更新 dp[mid+1]~dp[r]
假设我们现在已有了dp[l]~dp[mid]的上凸壳
那么如果保证 mid+1~r的斜率单调
就可以在线性时间内完成 dp[l]~dp[mid] 对 dp[mid+1]~dp[r]的更新
斜率单调可以在归并排序之前的排序预处理中解决
仅剩的问题:如何得到dp[l]~dp[mid] 构成的上凸壳?
在每一层最后归并的时候,我们用线形的时间使l~mid 的点 有序
即横坐标为第一关键字,纵坐标为第二关键字 升序排列,
那么处理完l~mid之后,
会得到l~mid 所有的点 升序排列的结果
对于一些有序的点,用单调栈维护斜率递减即可维护出上凸壳
然后利用这个上凸壳去更新mid+1~r 即可
复杂度为 O(nlogn)
#include<cmath>
#include<cstdio>
#include<algorithm>
#include<iostream> using namespace std; #define N 100001 const double eps=1e-; struct node
{
double A,B,R;
double X,Y;
double K;
int id; bool operator < (node p) const
{
/*if(fabs(X-p.X)>eps) return X<p.X;
return Y<p.Y;*/
return X<p.X;
} }e[N],t[N]; double dp[N],ans; int st[N],top; bool down(int i,int j,int k)
{
return (e[j].X-e[i].X)*(e[k].Y-e[i].Y)-(e[k].X-e[i].X)*(e[j].Y-e[i].Y)<;
} void solve(int l,int r)
{
if(l==r)
{
dp[l]=max(dp[l],dp[l-]);
e[l].Y=dp[l]/(e[l].A*e[l].R+e[l].B);
e[l].X=e[l].Y*e[l].R;
return;
}
int mid=l+r>>;
int opl=l,opr=mid+;
for(int i=l;i<=r;++i)
if(e[i].id<=mid) t[opl++]=e[i];
else t[opr++]=e[i];
for(int i=l;i<=r;++i) e[i]=t[i];
solve(l,mid);
top=;
for(int i=l;i<=mid;++i)
{
while(top> && !down(st[top-],st[top],i)) top--;
st[++top]=i;
}
int now=;
for(int i=mid+;i<=r;++i)
{
while(now<top && (e[st[now]].Y-e[st[now+]].Y)<(e[st[now]].X-e[st[now+]].X)*e[i].K) now++;
dp[e[i].id]=max(dp[e[i].id],e[st[now]].X*e[i].A+e[st[now]].Y*e[i].B);
}
solve(mid+,r);
opl=l; opr=mid+;
for(int i=l;i<=r;++i)
if(opl>mid) t[i]=e[opr++];
else if(opr>r) t[i]=e[opl++];
else if(e[opl]<e[opr]) t[i]=e[opl++];
else t[i]=e[opr++];
for(int i=l;i<=r;++i) e[i]=t[i];
} bool cmp(node p,node q)
{
return p.K>q.K;
} int main()
{
int n;
double m;
scanf("%d%lf",&n,&dp[]);
for(int i=;i<=n;++i)
{
scanf("%lf%lf%lf",&e[i].A,&e[i].B,&e[i].R);
e[i].K=-e[i].A/e[i].B;
e[i].id=i;
}
sort(e+,e+n+,cmp);
solve(,n);
printf("%.3lf",dp[n]);
}
1492: [NOI2007]货币兑换Cash
Time Limit: 5 Sec Memory Limit: 64 MB
Submit: 5830 Solved: 2342
[Submit][Status][Discuss]
Description
.png)
.png)
Input
Output
只有一个实数MaxProfit,表示第N天的操作结束时能够获得的最大的金钱数目。答案保留3位小数。
Sample Input
1 1 1
1 2 2
2 2 3
Sample Output
HINT
bzoj千题计划237:bzoj1492: [NOI2007]货币兑换Cash的更多相关文章
- [BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化)
[BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化) 题面 分析 dp方程推导 显然,必然存在一种最优的买卖方案满足:每次买进操作使用完所有的人民币:每次卖出操作卖出所有 ...
- [BZOJ1492][NOI2007]货币兑换Cash(斜率优化+CDQ分治)
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 5838 Solved: 2345[Submit][Sta ...
- BZOJ1492: [NOI2007]货币兑换Cash 【dp + CDQ分治】
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MB Submit: 5391 Solved: 2181 [Submit][S ...
- bzoj1492[NOI2007]货币兑换Cash cdq分治+斜率优化dp
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 5541 Solved: 2228[Submit][Sta ...
- [BZOJ1492] [NOI2007]货币兑换Cash 斜率优化+cdq/平衡树维护凸包
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 5907 Solved: 2377[Submit][Sta ...
- bzoj千题计划300:bzoj4823: [Cqoi2017]老C的方块
http://www.lydsy.com/JudgeOnline/problem.php?id=4823 讨厌的形状就是四联通图 且左右各连一个方块 那么破坏所有满足条件的四联通就好了 按上图方式染色 ...
- bzoj千题计划196:bzoj4826: [Hnoi2017]影魔
http://www.lydsy.com/JudgeOnline/problem.php?id=4826 吐槽一下bzoj这道题的排版是真丑... 我还是粘洛谷的题面吧... 提供p1的攻击力:i,j ...
- bzoj千题计划280:bzoj4592: [Shoi2015]脑洞治疗仪
http://www.lydsy.com/JudgeOnline/problem.php?id=4592 注意操作1 先挖再补,就是补的范围可以包含挖的范围 SHOI2015 的题 略水啊(逃) #i ...
- bzoj千题计划177:bzoj1858: [Scoi2010]序列操作
http://www.lydsy.com/JudgeOnline/problem.php?id=1858 2018 自己写的第1题,一遍过 ^_^ 元旦快乐 #include<cstdio> ...
随机推荐
- C#_委托
委托属于C#中的新名词,它的应用也非常广泛,例如事件就是委托最简单而又直接的例子. 那么首先说说什么是委托,其实委托在用过C或者C++的人看来就是函数指针,不过使用C#的大多数人都没有用过这两门语言, ...
- Js_图片切换左右点击
<!DOCTYPE HTML><html><head><meta http-equiv="Content-Type" content=&q ...
- DevOps on AWS之Cloudformation概念介绍篇
Cloudformation的相关概念 AWS cloudformation是一项典型的(IAC)基础架构即代码服务..通过编写模板对亚马逊云服务的资源进行调用和编排.借助cloudformation ...
- cf 1029D
题面 题目描述 给定含n个整数的数组a. 规定数x,y的合并为xy.如:数12与数3456的合并为数123456. 有数组中的位置对(i,j)(i≠j),计算使ai,aj的合并能被k整除的位置对数量. ...
- 三种迭代Java ArrayList方法及比较
闲来无事,研究一下Java Collection,首先是ArrayList. 通过三种方式遍历了长度为100000的ArrayList. import java.util.*; public clas ...
- 《Linux内核设计与实现》第一二章读书笔记
第一章 Linux内核简介 1.Unix简介 (一)概念:支持抢占式多任务.多进程.虚拟内存.换页.动态链接和TCP/IP网络的现代化操作系统. (二)Unix特点(层次化结构): Unix很简洁,仅 ...
- 20135337朱荟潼 Linux第二周学习总结——操作系统是如何工作的
一.计算机是如何工作的--总结 三个法宝 存储程序计算机.函数调用堆栈.中断机制 二.堆栈 1.是c语言程序运行时必须的一个记录调用路径和参数的空间. 函数调用框架.传递参数.保存返回地址.提供局部变 ...
- RYU 灭龙战 fourth day (1)
RYU 灭龙战 fourth day (1) 前言 对于流量的监控,对于一个网络管理人员来说是非常重要的,可以从可视化的角度,方便检测出哪里的设备出了问题:而在传统网络中,如果是哪里的设备出了问题的话 ...
- mosquitto集群配置
--------------------------------------------------------前言------------------------------------------ ...
- [转帖]win10 .Net Runtime Optimization Service占用大量CPU资源解决方法
win10 .Net Runtime Optimization Service占用大量CPU资源解决方法 https://blog.csdn.net/cwg2552298/article/detail ...