用途

废话,当然是在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学习笔记的更多相关文章

  1. Lightning Conductor 洛谷P3515 决策单调性优化DP

    遇见的第一道决策单调性优化DP,虽然看了题解,但是新技能√,很开森. 先%FlashHu大佬,反正我是看了他的题解和精美的配图才明白的,%%%巨佬. 废话不多说,看题: 题目大意 已知一个长度为n的序 ...

  2. CF868F Yet Another Minimization Problem 分治决策单调性优化DP

    题意: 给定一个序列,你要将其分为k段,总的代价为每段的权值之和,求最小代价. 定义一段序列的权值为$\sum_{i = 1}^{n}{\binom{cnt_{i}}{2}}$,其中$cnt_{i}$ ...

  3. 2018.09.28 bzoj1563: [NOI2009]诗人小G(决策单调性优化dp)

    传送门 决策单调性优化dp板子题. 感觉队列的写法比栈好写. 所谓决策单调性优化就是每次状态转移的决策都是在向前单调递增的. 所以我们用一个记录三元组(l,r,id)(l,r,id)(l,r,id)的 ...

  4. [BZOJ4850][JSOI2016]灯塔(分块/决策单调性优化DP)

    第一种方法是决策单调性优化DP. 决策单调性是指,设i>j,若在某个位置x(x>i)上,决策i比决策j优,那么在x以后的位置上i都一定比j优. 根号函数是一个典型的具有决策单调性的函数,由 ...

  5. BZOJ2216 Poi2011 Lightning Conductor 【决策单调性优化DP】

    Description 已知一个长度为n的序列a1,a2,...,an. 对于每个1<=i<=n,找到最小的非负整数p满足 对于任意的j, aj < = ai + p - sqrt( ...

  6. 决策单调性优化dp 专题练习

    决策单调性优化dp 专题练习 优化方法总结 一.斜率优化 对于形如 \(dp[i]=dp[j]+(i-j)*(i-j)\)类型的转移方程,维护一个上凸包或者下凸包,找到切点快速求解 技法: 1.单调队 ...

  7. 算法学习——决策单调性优化DP

    update in 2019.1.21 优化了一下文中年代久远的代码 的格式…… 什么是决策单调性? 在满足决策单调性的情况下,通常决策点会形如1111112222224444445555588888 ...

  8. BZOJ4899: 记忆的轮廓【概率期望DP】【决策单调性优化DP】

    Description 通往贤者之塔的路上,有许多的危机. 我们可以把这个地形看做是一颗树,根节点编号为1,目标节点编号为n,其中1-n的简单路径上,编号依次递增, 在[1,n]中,一共有n个节点.我 ...

  9. 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 ...

随机推荐

  1. 2019杭电多校三 C. Yukikaze and Demons (点分治)

    大意: 给定树, 每个点有一个十进制数位, 求有多少条路径组成的十进制数被$k$整除. 点分治, 可以参考CF715C, 转化为求$10^a x+b\equiv 0(mod\space k)$的$x$ ...

  2. hdu 2102 a计划问题。。 双层dfs问题

    Problem Description 可怜的公主在一次次被魔王掳走一次次被骑士们救回来之后,而今,不幸的她再一次面临生命的考验.魔王已经发出消息说将在T时刻吃掉公主,因为他听信谣言说吃公主的肉也能长 ...

  3. Spark机器学习基础-特征工程

    对连续值处理 0.binarizer/二值化 from __future__ import print_function from pyspark.sql import SparkSession fr ...

  4. win7用驱动精灵安装了bcm94352ac蓝牙驱动后还是不能用蓝牙的解决方法

    驱动精灵安装了驱动后,设备管理器处显示Bluetooth USB,但是没法用蓝牙,找不到蓝牙图标,后来在华硕官方下载了win7的Broadcom 蓝牙驱动程序装上之后就好了

  5. Python基础数据类型(数字、字符串、布尔、列表、元组、字典、集合等)

    数据类型 计算机顾名思义就是可以做数学计算的机器,因此,计算机程序理所当然地可以处理各种数值.但是,计算机能处理的远不止数值,还可以处理文本.图形.音频.视频.网页等各种各样的数据,不同的数据,需要定 ...

  6. 【转载】Sqlserver使用Group By进行分组并计算每个组的数量

    在SQL语句查询中,Group By语句时常用来进行分组操作,有时候在分组的同时还需要计算出每个组的数量多少.在Sqlserver数据库中可以使用Group By加Count聚合函数来实现此功能,即通 ...

  7. 关于Vue中,使用watch同时监听两个值的实现方法

    1. 先在computed中,用需要监听的两个值(start.end)定义一个对象(dateRange) computed: { dateRange () { const { start, end } ...

  8. Django中ORM多对多表的操作

    自己创建第三张表建立多对多关系 表的创建 # 老师表和学生表可以是一个多对多的关系,建表时可以手动建立第三张表建立关联 class Student(models.Model): name = mode ...

  9. 清楚html和css标签自带默认样式

    html, body, div, span, applet, object, iframe, h1, h2, h3, h4, h5, h6, p, blockquote, pre, a, abbr, ...

  10. nodejs库express是如何接收inbound json请求的

    这样几行简单的代码创建一个web服务器: var express = require('express'); var app = express(); var server = require('ht ...