题意理解
合唱队的站队要求是站中间的人高,站两边的人低,当一排人高矮随机站,问最少抽出几人,剩下的人是合唱队阵型?
问题分析
穷举法
合唱队阵型的要点是有一个峰顶,穷举合唱队每个人作为峰顶,看每种情况下满足合唱队阵型的最大人数有多少。
合唱队阵型分成两部分,以峰顶元素为轴,左边是递增序列,右边是递减序列,而递减序列也可以看成原始站队的翻转的递增序列。这样算法思路就是:对原始序列,计算每个元素以自身结尾的子序列的最大递增序列的长度(LIS),再对原始序列翻转,再次计算每个元素以自身结尾的子序列的最大递增序列的长度(LIS),所得的LIS子序列翻转就得到原始序列的递减序列的长度。将递增序列和递减序列相加再减1(递增和递减序列都考虑了峰值元素,所以减掉一个重复值),就得到每个元素为峰值的合唱队阵型的元素个数。取这个序列的最大值就是合唱队阵型的最大人数
总人数减去最大人数就是需要抽出的最少人数。
其他
这题没有想出来,思路有点奇特。
最大递增子序列的算法很多,这个是N^2的算法。
这题难在如何和LIS联系起来,阵型分成递增和递减的结合。
算法考试这题没想出来,现在算是还了这个债。一步一步往上爬。
链接
https://github.com/xierensong/learngit/blob/master/ncode/18/18.cpp