前言
2020暑假,回归OI。这两个dp难度比较低,但有一定价值,写个归纳总结。
线性dp小总结(优化)
洛谷P4933 大师
看懂题目后,容易列出dp方程:
- (f(i,j)) 表示以 (i) 结尾,公差为 (j) 的等差数列方案数。
[f(i,j) = sum_{{k | h_i - h_k = j}} f(k,j)
]
状态是 (O(n^2)) 的,转移是 (O(n)) 的,我们必须有一些优化。发现转移就是要对 (i) 前面所有满足 (h_k = h_i - j) 的 (f(k,j)) 求和。
于是我们枚举公差 (j):
-
令函数 (g(x) = sum_{{k|h_k=x,k<i}} f(k,j))
-
则有 (f(i,j) = g(h_i-j))
(g(x)) 可以在dp的过程中累积,转移便优化成了 (O(1))
总结:对于dp转移过程中如果遇见了求和,试着发现求和式子的规律,设计函数(并且该函数可以在dp过程中累积),从而降低转移的复杂度。
洛谷 P5858 「SWTR-03」Golden Sword
看懂题目后,容易列出dp方程:
- (f(i,j)) 表示以 (i) 结尾,当前原材料为 (j) 的最大耐久度
[f(i,j) = max_{j-1 le k le j+s} { f(i-1,k) } + a_i*j
]
状态是 (O(n^2)),转移是 (O(n)),我们必须有一些优化。试着模拟 (j) 的枚举,发现 (j) 在 (0 le j le w) 移动的过程中 ([j-1,j+s]) 也在向右移动。我们需要的就是这个移动区间的最大值。至此,我们发现这个转移可以使用单调队列实现 (O(1)) 转移。
如果在dp转移过程中遇见了求最值,试着发现最值区间的变化规律,尝试利用数据结构优化,从而降低转移的复杂度。