BZOJ3963: [WF2011]MachineWorks 【CDQ+斜率优化DP】*
BZOJ3963: [WF2011]MachineWorks
Description
你是任意性复杂机器公司(Arbitrarily Complex Machines, ACM)的经理,公司使用更加先进的机械设备生产先进的机器。原来的那一台生产机器已经坏了,所以你要去为公司买一台新的生产机器。你的任务是在转型期内尽可能得到更大的收益。在这段时间内,你要买卖机器,并且当机器被ACM公司拥有的时候,操控这些机器以获取利润。因为空间的限制,ACM公司在任何时候都只能最多拥有一台机器。
在转型期内,有若干台可能卖出的机器。作为先进机器的专家,对于每台机器Mi,你已经知道了其价格Pi和可以买入的日期Di。注意,如果不在第Di天买入机器Mi,那么别的人也会买走这一台机器,也就是说,以后你将没有机会购买这台机器了。如果ACM的钱低于一台机器的价格,那么你显然不可能买到这一台机器。
如果你在第Di天买入了机器Mi,那么ACM公司可以从第(Di)+1天开始使用这一台机器。每使用这台机器一天,就可以为公司创造出Gi美元的收益。
你可以决定要在买入之后的某一天,以一定的折扣价卖出这一台机器。收购市场对于每一台机器,都有一个折扣价Ri。你不能在卖出的那一天使用机器,但是你可以在卖出的那一天再买入一台新的。
在转型期结束后,ACM公司会卖掉当前所拥有的机器。你的任务就是最大化转型期间ACM公司可以得到的收入。
Input
输入包含若干组测试用例。每一组测试用例的第一行有3个正整数N,C和D。N是将会卖出的机器的台数(N<=105)(N<=10^5)(N<=105),C是在转型期开始时公司拥有的美元数量(C<=109)(C<=10^9)(C<=109),D是转型期持续的天数(D<=109)(D<=10^9)(D<=109)。
之后的N行每一行描述了一台机器的情况。每一行有4个正整数Di,Pi,Ri和Gi,分别表示这台机器卖出的时间,购买这台机器需要的美元数量,卖出这台机器的折扣价和使用这台机器可以得到的利润。这些数字满足1<=Di<=D,1<=Ri<Pi<=1091<=Di<=D,1<=Ri<Pi<=10^91<=Di<=D,1<=Ri<Pi<=109且1<=Gi<=1091<=Gi<=10^91<=Gi<=109.
最后一组测试用例后面的一行由3个0组成,表示输入数据。
Output
对于每一组测试用例,输出测试用例的编号,之后给出ACM公司在第D+1天结束后可以得到的最大数量的美元。
请依照下面给出的样例输出。
Sample Input
6 10 20
6 12 1 3
1 9 1 2
3 2 1 2
8 20 5 4
4 11 7 4
2 10 9 1
0 0 0
Sample Output
Case 1: 44
比较裸的CDQ+斜率优化吧
直接维护就行了?雾
首先定义一下DP:dpidp_idpi表示卖完第i台机器的最大收益,也就是卖出上一台机器的收益-pip_ipi
然后考虑转移dpi=max(dpj+rj+(di−dj−1)∗dj)−pidp_i=max(dp_j+r_j+(d_i-d_j-1)*d_j)-p_idpi=max(dpj+rj+(di−dj−1)∗dj)−pi
把中间展开dpi=dpj+rj−(dj+1)∗gj+di∗gj−pidp_i=dp_j+r_j-(d_j+1)*g_j+d_i*g_j-p_idpi=dpj+rj−(dj+1)∗gj+di∗gj−pi
然后变成点斜式dpi+pi−di∗gj=dpj+dj−(dj+1)∗gjdp_i+p_i-d_i*g_j=dp_j+d_j-(d_j+1)*g_jdpi+pi−di∗gj=dpj+dj−(dj+1)∗gj
然后我们只需要判断一下可不可行dpi>pidp_i>p_idpi>pi然后直接CDQ分治,用左边的维护凸包,然后将右边的斜率排一下序直接查找就好了
不想写归并所以就直接sort了,多个log影响不大
#include<bits/stdc++.h>
using namespace std;
#define LL long long
#define pi pair<LL,LL>
#define N 100010
#define INFF 1e16
#define INFI 0x3f3f3f3f3f3f3f3f
LL n,m,c;
LL dp[N];//买入机器i之后的最大收益
struct Machine{
LL d,p,r,g;
void init(){scanf("%lld%lld%lld%lld",&d,&p,&r,&g);}
}t[N];
struct Node{
LL x,y;
Node(){}
Node(int id){
x=t[id].g;
y=dp[id]+t[id].r-t[id].g*(t[id].d+);
}
Node(LL _x,LL _y){x=_x,y=_y;}
};
vector<Node> lq,rq;
deque<Node> q;
bool cmpMachine(Machine a,Machine b){return a.d<b.d;}
bool cmpNode(Node a,Node b){
if(a.x==b.x)return a.y<b.y;
return a.x<b.x;
}
double getk(Node a,Node b){
if(a.x==b.x)return INFF;
return (double) (a.y-b.y)/(double) (a.x-b.x);
}
void solve(int l,int r){
if(l==r){
dp[l]=max(dp[l],1ll*c);dp[l]-=t[l].p;
if(dp[l]<)dp[l]=-INFI;
return;
}
int mid=(l+r)>>;
solve(l,mid);
lq.clear();rq.clear();q.clear();
for(int i=l;i<=mid;i++)if(dp[i]!=-INFI)lq.push_back(Node(i));
for(int i=mid+;i<=r;i++)rq.push_back(Node(t[i].d,i));
sort(lq.begin(),lq.end(),cmpNode);
sort(rq.begin(),rq.end(),cmpNode);
for(int i=;i<lq.size();i++){
while(q.size()>=&&getk(lq[i],q.back())>getk(q.back(),q[q.size()-]))q.pop_back();
q.push_back(lq[i]);
}
if(q.size()){
for(int i=;i<rq.size();i++){
while(q.size()>=&&getk(q[],q[])>-rq[i].x)q.pop_front();
dp[rq[i].y]=max(dp[rq[i].y],q.front().x*rq[i].x+q.front().y);
}
}
solve(mid+,r);
}
int main(){
int T=;
while(scanf("%lld%lld%lld",&m,&c,&n)&&(n||m||c)){
for(int i=;i<=m;i++)t[i].init(),dp[i]=-INFI;
t[++m]=(Machine){n+,,,};dp[m]=-INFI;
sort(t+,t+m+,cmpMachine);
solve(,m);
printf("Case %d: %lld\n",++T,dp[m]);
}
return ;
}
BZOJ3963: [WF2011]MachineWorks 【CDQ+斜率优化DP】*的更多相关文章
- bzoj3963[WF2011]MachineWorks cdq分治+斜率优化dp
3963: [WF2011]MachineWorks Time Limit: 30 Sec Memory Limit: 256 MBSubmit: 270 Solved: 80[Submit][S ...
- 【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 ...
- 【bzoj3672】[Noi2014]购票 斜率优化dp+CDQ分治+树的点分治
题目描述 给出一棵以1为根的带边权有根树,对于每个根节点以外的点$v$,如果它与其某个祖先$a$的距离$d$不超过$l_v$,则可以花费$p_vd+q_v$的代价从$v$到$a$.问从每个点到1花费 ...
- BZOJ1492:[NOI2007]货币兑换 (CDQ分治+斜率优化DP | splay动态维护凸包)
BZOJ1492:[NOI2007]货币兑换 题目传送门 [问题描述] 小Y最近在一家金券交易所工作.该金券交易所只发行交易两种金券:A纪念券(以下简称A券)和B纪念券(以下简称B券).每个持有金券的 ...
- [Noi2014]购票 斜率优化DP+可持久化凸包
貌似网上大部分题解都是CDQ分治+点分治然后再斜率优化DP,我貌似并没有用这个方法. 这一题跟这题有点像,只不过多了一个l的限制 如果说直接跑斜率优化DP,存储整个序列的话,显然是不行的,如图所示(图 ...
- 【学习笔记】动态规划—斜率优化DP(超详细)
[学习笔记]动态规划-斜率优化DP(超详细) [前言] 第一次写这么长的文章. 写完后感觉对斜优的理解又加深了一些. 斜优通常与决策单调性同时出现.可以说决策单调性是斜率优化的前提. 斜率优化 \(D ...
- 『摆渡车 斜率优化dp及总结』
摆渡车的题解我已经写过一遍了,在这里,这次主要从斜率优化的角度讲一下摆渡车,并总结一下斜率优化会出现的一些奇奇怪怪的错误. 摆渡车 Description 有 n 名同学要乘坐摆渡车从人大附中前往人民 ...
- bzoj-4518 4518: [Sdoi2016]征途(斜率优化dp)
题目链接: 4518: [Sdoi2016]征途 Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地 ...
- bzoj-1096 1096: [ZJOI2007]仓库建设(斜率优化dp)
题目链接: 1096: [ZJOI2007]仓库建设 Description L公司有N个工厂,由高到底分布在一座山上.如图所示,工厂1在山顶,工厂N在山脚.由于这座山处于高原内陆地区(干燥少雨),L ...
随机推荐
- 使用Sandcastle 基于代码注释生成接口文档
一. 工具下载: 1. Sandcastle:Sandcastle是微软官方的文档生成工具,下载地址:http://www.codeplex.com/Sandcastle 2. SHFBGuidedI ...
- 用gitolite搭建git server
在Ubuntu上测试安装一下git server,为后面项目的代码管理做准备.记录流水账如下, 中间关于git 命令的使用说明不做过多解释,需要了解的请google或者直接git help: 我用到了 ...
- Lucene 初步 之 HelloWorld
万恶的源头 HelloWorld 要完成lucene 的配置 需要几个jar包 (如果需要可以留言我私发) 创建索引API分析: 1. Directory: 类代表一个Lucene索引的位置,FSDi ...
- 堆 Heap
2018-03-01 20:38:34 堆(Heap)是可以用来实现优先的队列的数据结构,而不是堆栈. 若采用数组或者链表实现优先队列 若采用树的结构 如果采用二叉搜索树,那么每次删除,比如删除最大值 ...
- JSP XML 数据处理
JSP XML 数据处理 当通过HTTP发送XML数据时,就有必要使用JSP来处理传入和流出的XML文档了,比如RSS文档.作为一个XML文档,它仅仅只是一堆文本而已,使用JSP创建XML文档并不比创 ...
- 编码转换 Native / UTF-8 / Unicode
Native/Unicode Native 这是一个例子,this is a example Unicode 这是一个例子,this is a example Native/UTF-8 Nativ ...
- Spring Cloud实战
Spring Cloud实战(一)-Spring Cloud Config Server https://segmentfault.com/a/1190000006149891 https://seg ...
- linux bash shell中,单引号、 双引号,反引号(``)的区别及各种括号的区别
一.单引号和双引号 首先,单引号和双引号,都是为了解决中间有空格的问题. 因为空格在Linux中时作为一个很典型的分隔符,比如string1=this is astring,这样执行就会报错.为了避免 ...
- 51nod-1179-最大的gcd(数学)
1179 最大的最大公约数 题目来源: SGU 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 收藏 关注 给出N个正整数,找出N个数两两之间最大公约数的最大 ...
- 图片服务器(FastDFS)的搭建
1.1 什么是FastDFS FastDFS是用c语言编写的一款开源的分布式文件系统.FastDFS为互联网量身定制,充分考虑了冗余备份.负载均衡.线性扩容等机制,并注重高可用.高性能等指标,使用Fa ...