[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. [控件] 创建出条形间隔效果的背景LineBackgroundView

    创建出条形间隔效果的背景LineBackgroundView 效果: 使用: // // ViewController.m // LineBackgroundView // // Created by ...

  2. zabbix的日常监控-自动发现端口并监测(服务器开启的所有端口或监控指定端口)(十三)

    动批量检查agent开放的端口 注:此方法给监控磁盘IO(即十二)篇过程一样: 注释:如果服务器上的应用都是固定的,不会随机产生的都可以使用自动发现端口来监控:  如果服务器会随机出现端口且每次启动程 ...

  3. print(n) 和 while/for 并列的时候, print()只会打印出最后一个结果

    n=0while n <10: n+=1print(n) # print(n)放在while的外面和while并列的时候, 只会打印出最后一个结果 n=0while n <10: n+=1 ...

  4. python3 80行代码实现贪吃蛇

    上面是实现的截图,废话不说,直接开始说一下代码 pos = { 'UP': (-1,0), 'DOWN':(+1,0), 'LEFT':(0,-1), 'RIGHT':(0,+1), } curren ...

  5. python第十三课——嵌套循环

    2.嵌套循环: 概念:循环中再定义循环,称为嵌套循环: [注意]嵌套循环可能有多层,但是一般我们实际开发最多两层就可以搞定了(99%的情况) 格式: 1).while中套while常用 2).whil ...

  6. node(一)安装nodejs最新版到debian,ubuntu,mint系统

    从官网得到,测试可以使用,本机为linux mint18 官网原文链接在此 //     直接使用sudo apt install nodejs安装的版本较老,而且命令必须使用nodejs //   ...

  7. socket.io+angular.js+express.js做个聊天应用(二)

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/www19940501a/article/details/27585321 接着上一篇 我用的开发工具 ...

  8. Java基础加强之集合

    集合整体框架图 各集合框架的概述 1. Collection(常用List和Set,不常用Queue和Vector),单元素集合. 2. Map(常用HashMap和TreeMap,不常用HashTa ...

  9. 20165302 ch02 课下作业

    20165302 ch02 课下作业 作业内容 补充完成课上测试(不能只有截图,要有分析,问题解决过程,新学到的知识点) 完成教材 p97 2.96 2.97,要有完备的测试 发一篇相关内容的博客, ...

  10. mysql太多连接问题及解决方案

    不管是JavaEE开发还是其他,只要是Linux系统下安装的mysql,通常默认最大连接为270. 如果你的客户端连接超过这个数,通常要么是配置文件修改,或者是命令行修改,配置文件修改和命令行修改的区 ...