整理的算法模板合集: ACM模板
实际上是一个全新的精炼模板整合计划
E - Sorting Books
一排书架上有 n n n 本书排成一排,每本书上有一个颜色 a i a_i ai,你可以每次将一本书移动到书架的最右端,如果书架上的书,颜色相同的书都排到了一块,我们就认为他是漂亮的,请问将这个书架通过上面的那一种操作排成漂亮的书架,最少需要几次操作?
Solution
其实是超级简单的一道题 ~
首先根据题意,先不考虑最优解,我们直接全部往右乱扔就一定能满足,但是不一定是最优解,解决最优解问题,很明显要么贪心,要么DP,要么贪心 + DP。
首先考虑贪心策略,我们一上来最直观的感受就是,如果有一种颜色,没有完全在一块,中间参杂着其他颜色的书,但是这种颜色的书的数量非常非常的多,很明显我们就可以贪心地把这种颜色中间的书移走,这些这种颜色的书就会自动合并到一块,肯定比把这种颜色的很多很多的书一个一个丢到最右边要来的快,这一点很容易想到。
所以我们来尝试找一下,有没有这种颜色的书,或者好几种,只要区间不重叠就好,数量比它中间其他颜色的书的数量还要多,也就是找到若干个区间,但是直接找区间有点困难,题目只需要我们输出最少操作次数即可,所以在有了这个完美的贪心策略以后,我们考虑DP来模拟这个过程。
我们设 f [ i ] f[i] f[i] 表示 i ⋯ n i\cdots n i⋯n 区间里不需要移动的书的最大数量,很显然答案就是 n − f [ 1 ] n-f[1] n−f[1]。除去不需要移动的,把剩下的按照颜色的分类丢到右边就行了。
我们想要找到的是区间里书数量最多的颜色,并且可以是很多种区间不重叠的颜色,要先找区间,所以我们先存一下每种颜色的出现的左右区间 l i l_i li 和 r i r_i ri ,因为要找数量最多的那一种,所以我们再使用 c n t cnt cnt 数组存一下每种颜色出现的次数。因为现在考虑的是逆序 DP,所以再存一下 c n t _ p o s t i \text cnt\_post_i cnt_posti ,表示的是 i ⋯ n i\cdots n i⋯n 之间每种颜色的书的数量。
考虑转移方程:
(1) f [ i ] = max { f [ i ] , f [ i + 1 ] } \tt f\ [i] = \max\{f\ [i], f[i + 1]\} f [i]=max{ f [i],f[i+1]}
(2) f [ i ] = max { f [ i ] , c n t _ p o s t [ a [ i ] ] } \tt f[i] = \max\{f [i], {cnt\_post} [ a[ i]]\} f[i]=max{ f[i],cnt_post[a[i]]}
我们取 i ⋯ n i\cdots n i⋯n 里当前最优的一种颜色,也就是只选择一个区间的情况。为什么不能写成 f [ i ] = max { f [ i ] , c n t _ p o s