LeetCode每日一题——854. 相似度为 K 的字符串

简介: 给你两个字母异位词 s1 和 s2 ,返回 s1 和 s2 的相似度 k 的最小值。

题目

对于某些非负整数 k ,如果交换 s1 中两个字母的位置恰好 k 次,能够使结果字符串等于 s2 ,则认为字符串 s1 和 s2 的 相似度为 k 。

给你两个字母异位词 s1 和 s2 ,返回 s1 和 s2 的相似度 k 的最小值。

示例

示例 1:

输入:s1 = “ab”, s2 = “ba”

输出:1

示例 2:

输入:s1 = “abc”, s2 = “bca”

输出:2

提示:

1 <= s1.length <= 20

s2.length == s1.length

s1 和 s2 只包含集合 {‘a’, ‘b’, ‘c’, ‘d’, ‘e’, ‘f’} 中的小写字母

s2 是 s1 的一个字母异位词

思路

1、自己想的是无剪枝和去重的暴力DFS和BFS搜索,但是结果都超时了

2、参考了官方题解,找到了对BFS剪枝和去重的内容

去重:用哈希表记录每次计算时得到的中间状态,当通过交换时发现当前状态已经计算过,则此时我们可以直接跳过该状态。

剪枝:我们只考虑s1[j] = s2[i] != s2[j]的情况,因为第i位后面如果第j位s1和s2已经相等,我们再做替换是多余的操作。

题解

暴力DFS超时代码

class Solution:
    def kSimilarity(self, A, B):
        ans = []
        s1 = list(A)
        def process(s1,index,s):
            if ''.join(s1) == B:
                ans.append(s)
                return
            for i in range(index, len(s1)):
                for j in range(i+1, len(s1)):
                    s1[i],s1[j] = s1[j], s1[i]
                    process(s1, i+1, s+1)
                    s1[i],s1[j] = s1[j], s1[i]
        process(s1, 0, 0)
        return min(ans)

BFS+剪枝+去重

class Solution:
    def kSimilarity(self, s1: str, s2: str) -> int:
      # 已走步长
        step, n = 0, len(s1)
        # 列表模拟队列,哈希表去重
        q, vis = [(s1, 0)], {s1}
        while True:
            tmp = q
            q = []
            # 遍历队列
            for s, i in tmp:
                if s == s2:
                    return step
                while i < n and s[i] == s2[i]:
                    i += 1
                for j in range(i + 1, n):
                    if s[j] == s2[i] != s2[j]:  # 剪枝,只在 s[j] != s2[j] 时去交换
                        t = list(s)
                        t[i], t[j] = t[j], t[i]
                        t = ''.join(t)
                        # 判重
                        if t not in vis:
                            vis.add(t)
                            q.append((t, i + 1))
            step += 1
目录
相关文章
|
7月前
|
Go 索引
【LeetCode 热题100】394:字符串解码(详细解析)(Go语言版)
本文详细解析了 LeetCode 热题 394:字符串解码。题目要求对编码字符串如 `k[encoded_string]` 进行解码,其中 `encoded_string` 需重复 `k` 次。文章提供了两种解法:使用栈模拟和递归 DFS,并附有 Go 语言实现代码。栈解法通过数字栈与字符串栈记录状态,适合迭代;递归解法则利用函数调用处理嵌套结构,代码更简洁。两者时间复杂度均为 O(n),但递归需注意栈深度问题。文章还总结了解题注意事项及适用场景,帮助读者更好地掌握字符串嵌套解析技巧。
190 6
|
8月前
|
存储 机器学习/深度学习 缓存
🚀 力扣热题 394:字符串解码(详细解析)(Go语言版)
文章提供了两种解法:栈结构和递归解法。栈解法通过维护数字栈与字符串栈,依次处理 `[` 和 `]`,构造解码结果;递归解法则利用函数调用逐层解析嵌套结构。两者时间复杂度均为 $O(n)$,空间复杂度也为 $O(n)$。栈解法直观易懂,适合初学者;递归解法优雅简洁,适合处理深度嵌套规则。掌握这两种方法,可灵活应对类似问题,提升解题能力。
273 11
|
C++
Leetcode第43题(字符串相乘)
本篇介绍了一种用C++实现的字符串表示的非负整数相乘的方法,通过逆向编号字符串,将乘法运算转化为二维数组的累加过程,最后处理进位并转换为字符串结果,解决了两个大数相乘的问题。
114 9
|
JavaScript
力扣3333.找到初始输入字符串Ⅱ
【10月更文挑战第9天】力扣3333.找到初始输入字符串Ⅱ
152 1
|
算法 C++
Leetcode第八题(字符串转换整数(atoi))
这篇文章介绍了LeetCode上第8题“字符串转换整数(atoi)”的解题思路和C++的实现方法,包括处理前导空格、正负号、连续数字字符以及整数溢出的情况。
171 0
【LeetCode 22】459.重复的子字符串
【LeetCode 22】459.重复的子字符串
127 0
【LeetCode 20】151.反转字符串里的单词
【LeetCode 20】151.反转字符串里的单词
116 0
【LeetCode 19】541.反转字符串II
【LeetCode 19】541.反转字符串II
98 0
【LeetCode 18】6.2.反转字符串
【LeetCode 18】6.2.反转字符串
86 0
|
存储 算法
LeetCode第43题字符串相乘
LeetCode第43题"字符串相乘"的解题方法,通过使用数组存储乘积并处理进位,避免了字符串转换数字的复杂性,提高了算法效率。
LeetCode第43题字符串相乘

热门文章

最新文章