YACS 2023年5月月赛 乙组 T1 升序排列(二) 题解
一道诈骗题,英语课上一直在想然后想出来了(
正难则反,我们很难按照题目所说的得到最少步数,可以考虑从排好序的状态开始。
这样,每次就从首或尾中选择一个移到任意一个位置了,简单了些。
(因为我们只能移动首尾,当前移动到哪里最优貌似可以贪心)
所以下面令初始状态为有序的那个,目标状态为初始时给定的那个。
接着,移动完成的目标就是初始的数组,不是很能看出来,也不太好搞,就再化简了一些。
按照题目所说,给的排列是
把初始状态中的每个数
显然,这不会影响答案,因为我们接着再把
经过一番转换后,目标数组变成了有序的而初始无序了,操作变为首尾选一个放到任意位置的最少操作次数。
考虑这个问题如何求解,这是一个简单的贪心了。找到目标数组中的 最大连续上升序列,答案即为
为什么呢?我们可以把它前面后面的都移到正确的位置,而最大连续上升序列就不需要动了,严谨的证明我不会,全靠猜的(
代码很短,和拼接单词差不多了:):(甲组 T1 线段树打表简单不讲,T2 OEIS 乱搞,T3 插头 DP 值得一讲)
#include <iostream> using namespace std; int n, x, ans, cnt; int loc[100005]; int main () { cin >> n; for (int i = 1; i <= n; i ++) { cin >> x; loc[x] = i; } for (int i = 1; i <= n; i ++) { if (i == 1 || loc[i] > loc[i - 1]) ++ cnt; else { ans = max (ans, cnt); cnt = 1; } } ans = max (ans, cnt); cout << n - ans; return 0; }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· .NET10 - 预览版1新功能体验(一)