首页 > 其他 > 详细

斜率优化(CDQ分治):BZOJ 1492: [NOI2007]货币兑换Cash

时间:2016-03-22 19:24:38      阅读:346      评论:0      收藏:0      [点我收藏+]

Description

技术分享

Input

第一行两个正整数N、S,分别表示小Y 能预知的天数以及初始时拥有的钱数。 接下来N 行,第K 行三个实数AK、BK、RateK,意义如题目中所述

Output

只有一个实数MaxProfit,表示第N 天的操作结束时能够获得的最大的金钱 数目。答案保留3 位小数。

Sample Input

3 100
1 1 1
1 2 2
2 2 3

Sample Output

225.000

HINT

技术分享

测试数据设计使得精度误差不会超过10-7。
对于40%的测试数据,满足N ≤ 10;
对于60%的测试数据,满足N ≤ 1 000;
对于100%的测试数据,满足N ≤ 100 000;

 
 1 #include <iostream>
 2 #include <cstring>
 3 #include <cstdio>
 4 #include <algorithm>
 5 using namespace std;
 6 const double eps=1e-8;
 7 const int maxn=100010;
 8 struct Node{
 9     double a,b,r,x,y,k;
10     int id;
11 }d[maxn],t[maxn];
12 double f[maxn];
13 int st[maxn],cnt,n;
14 double Abs(double a){return a>0?a:-a;}
15 double K(int a,int b){
16     if(Abs(d[a].x-d[b].x)<eps)return 1e20;
17     return (d[a].y-d[b].y)/(d[a].x-d[b].x);
18 }
19 void Solve(int l,int r){
20     if(l==r){
21         f[l]=max(f[l],f[l-1]);
22         d[l].y=f[l]/(d[l].r*d[l].a+d[l].b);
23         d[l].x=d[l].y*d[l].r;
24         return;
25     }
26     int mid=(l+r)>>1,t1=l,t2=mid+1;
27     for(int i=l;i<=r;i++){
28         if(d[i].id<=mid)
29             t[t1++]=d[i];
30         else 
31             t[t2++]=d[i];
32     }
33     for(int i=l;i<=r;i++)d[i]=t[i];
34     Solve(l,mid);
35     cnt=0;
36     for(int i=l;i<=mid;i++){
37         while(cnt>1&&K(i,st[cnt])-K(st[cnt],st[cnt-1])>=eps)cnt--;
38         st[++cnt]=i;
39     }
40     int fir=1;
41     for(int i=mid+1;i<=r;i++){
42         while(fir<cnt&&K(st[fir+1],st[fir])-d[i].k>=eps)fir++;
43         f[d[i].id]=max(f[d[i].id],d[st[fir]].x*d[i].a+d[st[fir]].y*d[i].b);
44     }
45     Solve(mid+1,r);
46     t1=l;t2=mid+1;
47     for(int i=l;i<=r;i++){
48         if(t2==r+1||(d[t2].x-d[t1].x>=eps||Abs(d[t2].x-d[t1].x)<eps&&d[t2].y-d[t1].y>=eps)&&t1<=mid)
49             t[i]=d[t1++];
50         else    
51             t[i]=d[t2++];
52     }    
53     for(int i=l;i<=r;i++)
54         d[i]=t[i];
55 }
56 
57 
58 bool cmp(Node a,Node b){
59     return a.k>b.k;
60 }
61 int main(){
62     scanf("%d%lf",&n,&f[1]);
63     for(int i=1;i<=n;i++){
64         scanf("%lf%lf%lf",&d[i].a,&d[i].b,&d[i].r);
65         d[i].k=-d[i].a/d[i].b;
66         d[i].id=i;
67     }
68     sort(d+1,d+n+1,cmp);
69     Solve(1,n);
70     printf("%.3lf\n",f[n]);
71     return 0;
72 }

 

斜率优化(CDQ分治):BZOJ 1492: [NOI2007]货币兑换Cash

原文:http://www.cnblogs.com/TenderRun/p/5307998.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!