[NOI2007]货币兑换

LG传送门

妥妥的\(n \log n\)cdq做法。

这题用cdq分治也可以\(n \log n\)但是在洛谷上竟然比一些优秀的splay跑得慢真是见了鬼了看来还是人丑常数大的问题

先推式子

(这一段与其他题解不会有太多不同,已经了解了的同学可以略过,注意一下转移中\(x\)和\(k\)表示什么就行了。)

设\(f[i]\)表示到第\(i\)天最多有多少钱,\(g[i]\)表示用第\(i\)天时的钱最多能买多少B券,易知\(g[i] = \frac {f[i]} {r[i] * a[i] + b[i]}\)。

得到转移:\(f[i] = \max \{ \max \limits _{j = 1} ^{i - 1} \{g[j] * \frac{b[i]} {a[i]} + r[j] * g[j]\} * a[i], f[i - 1] \}\),外面的\(\max\)可以单独判,里面的\(\max\)可以看出是一个斜率优化的式子(把\(\frac {b[i]} {a[i]}\)视作\(x\),把\(g[j]\)视作\(k\),把\(r[j] * g[j]\)视作\(b\))。但是我们发现斜率\(k\)并不是单调的,所以传统的斜率优化就无法解决这个问题了。

这时就衍生出两种写法了,一种是用splay维护凸包,一种是用cdq分治处理转移,我们要介绍的是后者。

考虑分治

对于任意一个\(f[i]\),我们只要考虑到所有\(1 \le j \le i - 1\)对它的影响就行了,cdq分治擅长处理这类问题。

对于一段区间\([l, r]\),先递归左子区间\([l, m]\),保证\([l, m]\)的\(f\)和\(g\)值都已经得到了;把左子区间按\(k\)递增排序,把右子区间按\(x\)递增排序,就可以按平时的斜率优化来\(O(n)\)转移;再把右子区间按在原序列中的位置递增排序,然后递归右子区间,此时左子区间对右子区间的影响都已经被考虑完了;边界是\(l == r\),到这里我们可以发现\(1\)到\(i - 1\)对\(i\)的影响都已经被考虑过了,别忘了\(f[i - 1]\)到\(f[i]\)的转移。

这样做是\(O(n (\log n) ^ 2)\)的,让人有点不爽,事实上我们可以做到\(O(n \log n)\)。

怎样做到1个\(\log\)

事实上cdq分治本身是一个归并的过程,我们可以利用这个过程去掉排序的复杂度。

我们希望拿到\([l, r]\)这个区间的时候\(x\)是单调的,于是在外面把原序列按\(x\)递增排序;拿到一个\(x\)递增的区间后,我们希望在原序列中靠左的东西去到左子区间,于是我们把\([l, r]\)扫一遍,把在原序列中位置\(\le m(m = l + r >> 1)\)的东西放左边,\(\ge m\)的放右边,而且左右子区间对于\(x\)的单调性没有受到影响;我们要处理左边对右边的影响,于是先递归左子区间,再像平时一样斜率优化处理转移,然后递归右子区间;我们希望一个区间的左子区间递归回来的时候是对于\(k\)单调递增的,于是在最后对\(k\)做一遍归并排序。这样每一层递归是\(O(n)\)的。

奉上蒟蒻的大常数代码。

//written by newbiechd
#include <iostream>
#include <iomanip>
#include <algorithm>
#define R register
#define I inline
#define D double
using namespace std;
const int N = 100003;
const D eps = 1e-8;
int q[N];
D f[N], g[N];
struct cash {
int id;
D a, b, r, x;
cash() {}
cash(int id, D a, D b, D r) : id(id), a(a), b(b), r(r), x(b / a) {}
I int operator < (cash q) { return x != q.x ? x < q.x : id < q.id; }
}p[N], b[N];
I D cross(int u, int v) { return (p[u].r * g[p[u].id] - p[v].r * g[p[v].id]) / (g[p[v].id] - g[p[u].id]); }
I D calc(int u, int v) { return g[p[u].id] * (p[v].x + p[u].r); }
I void update(int u, D v) {
if (f[p[u].id] < v)
f[p[u].id] = v, g[p[u].id] = f[p[u].id] / (p[u].b + p[u].r * p[u].a);
}
void solve(int l, int r) {
if (l == r) {
update(l, f[p[l].id - 1]);
return ;
}
R int m = (l + r) >> 1, i, h, t;
for (h = l, t = m + 1, i = l; i <= r; ++i)
p[i].id <= m ? b[h++] = p[i] : b[t++] = p[i];
for (i = l; i <= r; ++i)
p[i] = b[i];
solve(l, m), h = 1, t = 0;
for (i = l; i <= m; ++i) {
while (h < t && cross(q[t], i) < cross(q[t - 1], i) + eps)
--t;
q[++t] = i;
}
for (; i <= r; ++i) {
while (h < t && calc(q[h], i) < calc(q[h + 1], i) + eps)
++h;
update(i, calc(q[h], i) * p[i].a);
}
solve(m + 1, r);
for (h = l, t = m + 1, i = l; h <= m && t <= r; )
g[p[h].id] < g[p[t].id] ? b[i++] = p[h++] : b[i++] = p[t++];
while (h <= m)
b[i++] = p[h++];
while (t <= r)
b[i++] = p[t++];
for (i = l; i <= r; ++i)
p[i] = b[i];
}
int main() {
ios::sync_with_stdio(0);
R int n, i;
D a, b, r;
cin >> n >> f[1];
for (i = 1; i <= n; ++i)
cin >> a >> b >> r, p[i] = cash(i, a, b, r);
g[1] = f[1] / (p[1].r * p[1].a + p[1].b), sort(p + 1, p + n + 1),
solve(1, n), cout << fixed << setprecision(3) << f[n];
return 0;
}

[NOI2007]货币兑换 cdq分治,斜率优化的更多相关文章

  1. BZOJ_1492_[NOI2007]货币兑换Cash_CDQ分治+斜率优化

    BZOJ_1492_[NOI2007]货币兑换Cash_CDQ分治+斜率优化 Description 小Y最近在一家金券交易所工作.该金券交易所只发行交易两种金券:A纪念券(以下简称A券)和 B纪念券 ...

  2. BZOJ1492:[NOI2007]货币兑换 (CDQ分治+斜率优化DP | splay动态维护凸包)

    BZOJ1492:[NOI2007]货币兑换 题目传送门 [问题描述] 小Y最近在一家金券交易所工作.该金券交易所只发行交易两种金券:A纪念券(以下简称A券)和B纪念券(以下简称B券).每个持有金券的 ...

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

    [BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化) 题面 分析 dp方程推导 显然,必然存在一种最优的买卖方案满足:每次买进操作使用完所有的人民币:每次卖出操作卖出所有 ...

  4. 【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 ...

  5. bzoj1492[NOI2007]货币兑换Cash cdq分治+斜率优化dp

    1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 5541  Solved: 2228[Submit][Sta ...

  6. BZOJ 1492: [NOI2007]货币兑换Cash [CDQ分治 斜率优化DP]

    传送门 题意:不想写... 扔链接就跑 好吧我回来了 首先发现每次兑换一定是全部兑换,因为你兑换说明有利可图,是为了后面的某一天两种卷的汇率差别明显而兑换 那么一定拿全利啊,一定比多天的组合好 $f[ ...

  7. bzoj3672/luogu2305 购票 (运用点分治思想的树上cdq分治+斜率优化dp)

    我们都做过一道题(?)货币兑换,是用cdq分治来解决不单调的斜率优化 现在它放到了树上.. 总之先写下来dp方程,$f[i]=min\{f[j]+(dis[i]-dis[j])*p[i]+q[i]\} ...

  8. BZOJ 3963 HDU3842 [WF2011]MachineWorks cdq分治 斜率优化 dp

    http://acm.hdu.edu.cn/showproblem.php?pid=3842 写的check函数里写的<但是应该是<=,调了一下午,我是个zz. 就是普通的斜率优化因为有两 ...

  9. BZOJ4700 适者(贪心+cdq分治+斜率优化)

    首先考虑怎么安排攻击顺序.显然如果攻击了某台兵器就应该一直连续攻击直到将其破坏,破坏所需时间可以直接算出来,设其为b.假设确定了某个破坏顺序,如果交换相邻两个兵器,显然不会对其他兵器造成影响,两种顺序 ...

随机推荐

  1. 转: c#.net利用RNGCryptoServiceProvider产生任意范围强随机数的办法

    //这样产生0 ~ 100的强随机数(含100) ; int rnd = int.MinValue; decimal _base = (decimal)long.MaxValue; ]; System ...

  2. 转:未能打开编辑器:Unmatched braces in the pattern.

    原文地址:http://blog.csdn.net/hytdsky/article/details/4736462 Eclipse出现这个问题而不能查看源代码  原因就是语言包的问题 出现这个问题了 ...

  3. Python学习---Django关于POST的请求解析源码分析

    当有请求到来之后,先判断请求头content_type是不是[application/x-www-form-urlencoded] --> 如果是则将请求数据赋值给request.body然后解 ...

  4. MVC中JavaScript和CSS的自动打包与压缩

    在程序中安装System.Web.Optimization程序集 依赖关系如下图所示: 添加BundleConfiguration类 代码如下所示 注意必须使用对应的ScriptBundle和Styl ...

  5. ACE.js自定义提示实现方法

    ACE.js自定义提示实现方法 时间 2015-11-19 00:55:22  wsztrush's blog 原文  http://wsztrush.github.io/编程技术/2015/11/0 ...

  6. 【ORACLE】使用中注意事项(二)

    问题1:ORACLE在插入数据的时候,有时候中文变成????? 原因:由于当前计算机的字符集和服务器上的字符集不一致,导致中文乱码. 解决方案: 在当前使用的计算机中设置环境变量 在我的电脑上右键属性 ...

  7. Java虚拟机19:再谈四种引用状态

    JVM的四种引用状态 在Java虚拟机5:Java垃圾回收(GC)机制详解一文中,有简单提到过JVM的四种引用状态,当时只是简单学习,知道有这么一个概念,对四种引用状态理解不深.这两天重看虚拟机这部分 ...

  8. P1710 地铁涨价

    题目背景 本题开O2优化,请注意常数 题目描述 博艾市除了有海底高铁连接中国大陆.台湾与日本,市区里也有很成熟的轨道交通系统.我们可以认为博艾地铁系统是一个无向连通图.博艾有N个地铁站,同时有M小段地 ...

  9. Day4 数组

    双重for循环 外循环控制行,内循环控制列. //乘法表 ; i <= ; i++) { ; j <= i ;j++) { System.out.print(j+"*" ...

  10. c模拟银行家资源分配算法

    #include<stdio.h> #define PNUMBER 5//进程个数 #define SNUMBER 3//资源种类个数 //资源的种类,三种 char stype[SNUM ...