AGC007F 题解

题意

给定两个长为 \(n\) 的字符串 \(S, T\),求最少进行多少次操作才能使 \(S = T\)。

一次操作定义为:对于 \(i = 1, 2, .. n\),令第 \(i\) 位为操作 的第 \(i - 1\) 位或操作 的第 \(i\) 位。(\(i = 1\) 时只能第二种)

分析

赛时想了一个假的贪心,想到设 \(pos_i\) 为在答案中 \(T_i\) 是由 \(S_{pos_i}\) 贡献的,可以发现 \(\forall 1\le i < n, pos_i\le pos_{i + 1}\),而且若 \(i < j\) 有交,一定先操作完 \(j\)。

这个思路看上去没有问题,但是后面的结论是错的,当时我这样想的原因是:若 \(i\) 先操作完,\(j\) 就没有了,但真的是这样吗,可以 \(pos_i\) 先操作到 \(pos_j - 1\),\(pos_j\) 操作到 \(i + 1\) 及之后。

错误原因:并没有做到最优,能早点完成的操作放在了后面做。

Update:破案了,这道题明显是倒着做消除后效性,直接贪这么后效怎么做。倒着做好写多了/kel

考虑已经得到了 \(pos_i\)(对于 \(i\) 找第一个 \(j\) 使得 \(s_j = t_i\) 且 \(j \le pos_{i + 1}\)),相当于 \(S = 1, 2, ... n\),\(T = pos_1, pos_2, ..., pos_n\),每次我们尽量往右扩展,然后给有用的留 1 个让它往右扩展,发现这样一定是最优的。

考虑怎么维护这个,先把无用的去掉,把相同的 S 缩到一起,每次从右往左,依次进行 把当前区间缩到只有右端点,把右端点尽量向右扩展 的流程,这是好维护的。

考虑为什么对

  1. 可以根据贪心过程逆推操作
  2. 已经是最贪的了

考虑怎么维护,找规律,发现进行了 \(j\) 次操作后,当前区间 \(i\) 的 \(R_i = R_{i + j} - j\)。令 \(R_j(j > n) = +\inf\) 即可。

相关推荐
I_LPL1 天前
hot100贪心专题
数据结构·算法·leetcode·贪心
A923A4 天前
【洛谷刷题 | 第四天】
算法·前缀和·贪心·洛谷·差分
zaiyang遇见10 天前
[CSP-J 2020] 优秀的拆分
贪心·模拟·csp-j·信息学奥赛·程序设计竞赛·二进制拆分
I_LPL13 天前
hot 100 普通数组、矩阵专题
java·数据结构·矩阵·动态规划·贪心·数组·求职面试
zaiyang遇见1 个月前
[GESP202509 六级] 货物运输
深度优先·贪心·树的遍历·gesp六级·树的存储
不穿格子的程序员2 个月前
从零开始刷算法——贪心篇1:跳跃游戏1 + 跳跃游戏2
算法·游戏·贪心
chao_7892 个月前
跳跃游戏系列【贪心算法】
python·算法·游戏·贪心算法·贪心
Wuliwuliii3 个月前
Hall定理的应用
贪心·hall定理
长安er3 个月前
LeetCode121/55/45/763 贪心算法理论与经典题解析
java·数据结构·算法·leetcode·贪心算法·贪心
罗湖老棍子3 个月前
瑞瑞的木板(洛谷P1334 )
c++·算法·优先队列·贪心·哈夫曼树