决策单调性优化DP学习笔记
用途
废话,当然是在DP式子满足某些性质的时候来优化复杂度……
定义
对于\(j\)往大于\(j\)的\(i\)转移,可以表示成一个关于\(i\)的函数\(f_j(i)\),也就是\(dp_i=\max/\min\{f_j(i)\}\)。
若是取\(\max\),并且在某一个地方\(f_j(i)\)从下面跑到了\(f_k(i)\)的上面(如果加入\(f_j(i)\)这个函数时本来就在\(f_k(i)\)的上面那么不算),就称之为\(j\)取代了\(k\)。取\(\min\)同理。
如果满足决策单调性,那么必须满足:对于任意\(i,j\in [1,n]\),\(f_j(x)\)与\(f_i(x)\)没有交点或是只有一个交点,也就是\(i\)和\(j\)最多取代一次。
狭义的决策单调性是指对于\(i<j\)和他们的决策点\(p_i,p_j\),会有\(p_i<p_j\)。也就是说,对于\(j<k\),就只能有\(k\)取代\(j\)。
广义的决策单调性指只要一个题中取代关系唯一(只有大取代小或小取代大),那么都可以做。
显然,狭义决策单调性也被包含在广义决策单调性内。
显然,狭义和广义是我自己编出来的。
狭义决策单调性
分治
简单易懂,适合\(dp_i=\max/\min\{g_j+w_{i,j}\}\)这种与\(dp_j\)无关的递推式。
考虑一个类似整体二分的做法:设\(solve(l,r,L,R)\)表示已知\([l,r]\)全部由\([L,R]\)转移而来,求出\(dp[l,r]\)。
每次设\(mid=(l+r)/2\),然后暴力扫\([L,R]\),找到\(mid\)的决策点\(p\),然后分治\(solve(l,mid-1,L,p)\),\(solve(mid+1,r,p,R)\)。
复杂度:会做\(\log n\)层,每层的\(\sum(r-l+R-L)\)都是\(O(n)\)的,所以总复杂度\(n\log n\)。
优点:当\(w_{i,j}\)不能\(O(1)\)或\(O(\log n)\)计算时用暴力扫的方式或许可以做出来。
缺点:转移不按顺序,所以递推式必须与\(dp_j\)无关。
注意:如果做的时候还扫了\([R,l]\),那么复杂度分析就会挂!
单调队列
适合\(dp_i=\max/\min\{dp_j+w_{i,j}\}\)的递推式,要求\(w_{i,j}\)可以快速求出。
狭义决策单调性保证了比较靠前的决策点被取代后就可以被忽略了,所以可以维护一个队列,队首是当前最优的决策点,队尾是后面加入、还有潜力取代前面的决策点。(记\(q[l]\)为队首,\(q[r]\)为队尾)
每次考虑\(q[l]\)和\(q[l+1]\),如果\(q[l]\)没那么优,那么就把它弹掉。
加入当前点的时候,就要搞一些事情了。
如果什么都不管直接加入,那么会出现这样的尴尬情况:\(q[l]\)优于\(q[l+1]\),但\(q[l]\)劣于\(q[l+2]\)。
如何避免呢?使用上面\(f_j(x)\)函数数形结合的思想,如果加入这个点\(i\)后\(q[r]\)潜力就没了,也就是\(i\)取代\(q[r-1]\)比\(q[r]\)取代\(q[r-1]\)还快,那么就把\(q[r]\)弹掉。这样一来,上面的\(q[l+1]\)根本就不会出现在队列里面。
至于判断在哪里会取代,那就要二分那个位置了。
复杂度:显然\(O(n\log n)\)。
优点:转移按顺序。
缺点:要求\(w_{i,j}\)可以快速求出。
广义决策单调性
大取代小的时候上面已经讲过,不再赘述。
单调栈
现在是小取代大,比如\(f_2(i)\)一直默默无闻,而\(f_4(i)\)一直是统治地位,突然在\(i=10\)的时候2把4取代了。
这种时候,就不能再用单调队列,而是要用单调栈了。当栈顶没有下面的函数优的时候就弹栈。压入当前函数时若\(st[top-1]\)取代\(i\)比\(st[top]\)还快就弹栈。
本质上其实和单调队列差别不大,也是要二分,也是要求\(w_{i,j}\)能快速求出,复杂度也是\(O(n\log n)\)。
不过小取代大这种情况只有这一种做法,所以就没有优缺点一说了。
判断决策单调性
讲了这么多做法,还没有讲怎么判断这一性质。
一般来说应该是先写出DP式子,各种优化方法试一遍都不行,那就尝试严谨或感性证明。
如果实在不会证,那可能可以打表猜结论。猜中了当然好,但如果随机数据都过了,而结论其实是假的,那……
那你就要祈祷出题人用的就是随机数据。
一般来说我倾向于随机数据猜结论。
决策单调性优化DP学习笔记的更多相关文章
- Lightning Conductor 洛谷P3515 决策单调性优化DP
遇见的第一道决策单调性优化DP,虽然看了题解,但是新技能√,很开森. 先%FlashHu大佬,反正我是看了他的题解和精美的配图才明白的,%%%巨佬. 废话不多说,看题: 题目大意 已知一个长度为n的序 ...
- CF868F Yet Another Minimization Problem 分治决策单调性优化DP
题意: 给定一个序列,你要将其分为k段,总的代价为每段的权值之和,求最小代价. 定义一段序列的权值为$\sum_{i = 1}^{n}{\binom{cnt_{i}}{2}}$,其中$cnt_{i}$ ...
- 2018.09.28 bzoj1563: [NOI2009]诗人小G(决策单调性优化dp)
传送门 决策单调性优化dp板子题. 感觉队列的写法比栈好写. 所谓决策单调性优化就是每次状态转移的决策都是在向前单调递增的. 所以我们用一个记录三元组(l,r,id)(l,r,id)(l,r,id)的 ...
- [BZOJ4850][JSOI2016]灯塔(分块/决策单调性优化DP)
第一种方法是决策单调性优化DP. 决策单调性是指,设i>j,若在某个位置x(x>i)上,决策i比决策j优,那么在x以后的位置上i都一定比j优. 根号函数是一个典型的具有决策单调性的函数,由 ...
- BZOJ2216 Poi2011 Lightning Conductor 【决策单调性优化DP】
Description 已知一个长度为n的序列a1,a2,...,an. 对于每个1<=i<=n,找到最小的非负整数p满足 对于任意的j, aj < = ai + p - sqrt( ...
- 决策单调性优化dp 专题练习
决策单调性优化dp 专题练习 优化方法总结 一.斜率优化 对于形如 \(dp[i]=dp[j]+(i-j)*(i-j)\)类型的转移方程,维护一个上凸包或者下凸包,找到切点快速求解 技法: 1.单调队 ...
- 算法学习——决策单调性优化DP
update in 2019.1.21 优化了一下文中年代久远的代码 的格式…… 什么是决策单调性? 在满足决策单调性的情况下,通常决策点会形如1111112222224444445555588888 ...
- BZOJ4899: 记忆的轮廓【概率期望DP】【决策单调性优化DP】
Description 通往贤者之塔的路上,有许多的危机. 我们可以把这个地形看做是一颗树,根节点编号为1,目标节点编号为n,其中1-n的简单路径上,编号依次递增, 在[1,n]中,一共有n个节点.我 ...
- 2018.10.14 NOIP训练 猜数游戏(决策单调性优化dp)
传送门 一道神奇的dp题. 这题的决策单调性优化跟普通的不同. 首先发现这道题只跟r−lr-lr−l有关. 然后定义状态f[i][j]f[i][j]f[i][j]表示猜范围为[L,L+i−1][L,L ...
随机推荐
- Codeforces Round #563 Div. 2
A:显然排序即可. #include<bits/stdc++.h> using namespace std; #define ll long long #define inf 100000 ...
- MyBatis Generator 自动生成的POJO对象的使用(二)
四.Example Class使用说明 示例类指定如何构建动态where子句. 表中的每个非BLOB列都可以选择包含在where子句中. 示例是演示此类用法的最佳方法. 示例类可用于生成几乎无限制的w ...
- Spring Cloud Alibaba学习笔记(10) - Spring消息编程模型下,使用RocketMQ收发消息
编写生产者 集成 添加依赖 <dependency> <groupId>org.apache.rocketmq</groupId> <artifactId&g ...
- 谷歌浏览器调用activex控件方法
原文转自 https://jingyan.baidu.com/article/af9f5a2d0ebe5543140a4596.html activex是由微软开发,所以在支持上,目前原生态支持的只有 ...
- interface Part3(实现:显示和隐式)
1. 接口的实现实际上和类之间的继承是一样的,也是重写了接口中的方法,让其有了具体的实现内容. 2. 但需要注意的是,在类中实现一个接口时必须将接口中的所有成员都实现,否则该类必须声明为抽象类,并将接 ...
- 一、ribbon如何集成在openfeign中使用
所有文章 https://www.cnblogs.com/lay2017/p/11908715.html 正文 ribbon是springcloud封装的一个基于http客户端负载均衡的组件.spri ...
- 部署Java项目到阿里云服务器(Ubuntu16.04 64位)
生成Jar包 1.进入到项目所在的路径下,打开cmd命令控制台,使用如下命令打包项目. mvn package --前提将项目中使用的maven配置到系统的环境变量中 2.打包完成的jar包在项目目录 ...
- Nginx作为缓存服务
缓存类型 (1) 服务器缓存 服务端缓存一般使用Memcache.Redis (2)代理缓存 (3)客户端缓存 代理缓存流程图 第一步:客户端第一次向Nginx请求数据a: 第二步:当Nginx发现缓 ...
- Django :中间 件与csrf
一.中间件 什么是中间件 中间件有什么用 自定义中间件 中间件应用场景 二.csrf csrf token跨站请求伪造 一.中间件 1.什么是中间件 中间件顾名思义,是介于request与respon ...
- Go语言中的IO操作、Flag包以及urfave/cli命令行框架
一.格式化输入和输出 1.从终端获取用户的输入 fmt.Scanf 空格作为分隔符,占位符和格式化输出的一致 fmt.Scan 从终端获取用户的输入,存储在Scanln中的参数里,空格和换行符作为 ...