Leetcode 2911. Minimum Changes to Make K Semi-palindromes

  • [Leetcode 2911. Minimum Changes to Make K Semi-palindromes](#Leetcode 2911. Minimum Changes to Make K Semi-palindromes)
    • [1. 解题思路](#1. 解题思路)
    • [2. 代码实现](#2. 代码实现)

1. 解题思路

这一题属实也是把我坑惨了......

坦率地说,这道题本身并没有啥难度,但是坑爹的是题目表述简直有毒,有两个细节题目里面压根没提,一个是我从中文版本的题目当中发现的,另一个则是我根据失败的样例当中反推出来的,这简直有毒......

这道题本身思路上还是挺直接的,就是一个动态规划的题目,考虑每一种切分的方式,然后考察其中最小的变化次数即可。

然后对于每一个切分得到的子串,我们要求其变化所需的最小变化次数,我们只需要找到其所有对长度 l l l整除的 d d d,然后切分semi子串,再考察其中每一种分法下所需的变化次数之和,最后取最小值即可。

因此,我们就将上述题目拆分完成了,后续只要对其实现一下即可,测试发现是可以在有效时间内完成所有测试样例的。

但是,这里但是就来了,题目中遗漏了两个非常非常重要的说明,把我给坑惨了!!!

首先,这里semi-palindrome的定义事实上要求将其使用d进行切分后,每一个子串都得是回文 ,其次,题中也没有具体说,但是实际测试发现,这里对子串的切分要求每一个子串长度至少为2

这简直就是简直了!!!

到底谁出的题目啊,只能说,出来挨打!!!

2. 代码实现

给出python代码实现如下:

python 复制代码
class Solution:
    def minimumChanges(self, s: str, k: int) -> int:
        n = len(s)
        
        @lru_cache(None)
        def count(sub):
            cnt = 0
            n = len(sub)
            for i in range(n//2):
                if sub[i] != sub[n-1-i]:
                    cnt += 1
            return cnt
        
        @lru_cache(None)
        def count_change(s):
            if s == s[::-1]:
                return 0
            n = len(s)
            ans = count(s)
            for d in range(len(s)//2, 0, -1):
                if n % d != 0:
                    continue
                k = n // d
                cnt = 0
                for i in range(d):
                    sub = "".join([s[i+j*d] for j in range(k)])
                    cnt += count(sub)
                ans = min(ans, cnt)
            return ans
        
        @lru_cache(None)
        def dp(idx, k):
            if idx+2*k > n:
                return math.inf
            elif k == 1:
                return count_change(s[idx:])
            return min(count_change(s[idx:j]) + dp(j, k-1) for j in range(idx+2, n))
        
        ans = dp(0, k)
        return ans   

提交代码评测得到:耗时5650ms,占用内存71.8MB。

相关推荐
小猪咪piggy8 小时前
【算法】day16 动态规划
算法·动态规划
Bi_BIT3 天前
代码随想录训练营打卡Day38| 动态规划part06
算法·动态规划
利刃大大3 天前
【动态规划:01背包】01背包详解 && 模板题 && 优化
c++·算法·动态规划·力扣·背包问题
Miraitowa_cheems3 天前
LeetCode算法日记 - Day 94: 最长的斐波那契子序列的长度
java·数据结构·算法·leetcode·深度优先·动态规划
不染尘.4 天前
2025_11_5_刷题
开发语言·c++·vscode·算法·贪心算法·动态规划
闻缺陷则喜何志丹4 天前
【贪心 字典序 回文 最长公共前缀】LeetCode3734. 大于目标字符串的最小字典序回文排列|分数未知
c++·算法·力扣·贪心·字典序·回文·最长公共前缀
Jasmine_llq7 天前
《P6772 [NOI2020] 美食家》
动态规划·状态压缩·矩阵快速幂·事件排序与分段处理·图论基础(图的表示与遍历)
mifengxing8 天前
力扣每日一题——接雨水
c语言·数据结构·算法·leetcode·动态规划·
贝塔实验室8 天前
LDPC 码的构造方法
算法·fpga开发·硬件工程·动态规划·信息与通信·信号处理·基带工程
ゞ 正在缓冲99%…8 天前
leetcode1312.让字符串成为回文串的最少插入次数
数据结构·算法·leetcode·动态规划·记忆化搜索