Bootstrap

【经典算法】LeetCode 392 判断子序列(Java/C/Python3/Go实现含注释说明,Easy)

  • 标签(题目类型):双指针,字符串

题目描述

给定字符串 st,判断 s 是否为 t 的子序列。

你可以认为,如果 s 可以通过将 t 中的一些字符(也可以不删除)而不改变其余字符的相对位置来获得,那么 s 就是 t 的一个子序列。

原题:LeetCode 392

思路及实现

方式一:双指针法

思路

通过两个指针分别指向 st 的开头,然后逐个比较字符,如果相等则两个指针都向后移动一位,如果不等则只移动 t 的指针,直到 s 的指针指向末尾,则说明 st 的子序列。

代码实现

Java版本
public boolean isSubsequence(String s, String t) {
    int i = 0, j = 0;
    while (i < s.length() && j < t.length()) {
        if (s.charAt(i) == t.charAt(j)) {
            i++; // s的指针向后移动
        }
        j++; // t的指针始终向后移动
    }
    return i == s.length(); // 如果s的指针走到了末尾,说明s是t的子序列
}

说明:通过双指针 ij 分别指向 st 的当前字符,并在满足条件时移动指针。

C语言版本
bool isSubsequence(char *s, char *t) {
    int i = 0, j = 0;
    while (s[i] && t[j]) {
        if (s[i] == t[j]) {
            i++;
        }
        j++;
    }
    return !s[i]; // 如果s的指针走到了末尾(即s[i]为'\0'),说明s是t的子序列
}

说明:C语言通过字符数组和指针实现与Java类似的逻辑。

Python3版本
def isSubsequence(s: str, t: str) -> bool:
    i = j = 0
    while i < len(s) and j < len(t):
        if s[i] == t[j]:
            i += 1
        j += 1
    return i == len(s)

说明:Python代码与Java和C语言类似,只是语法有所不同。

Golang版本
func isSubsequence(s string, t string) bool {
    i, j := 0, 0
    for i < len(s) && j < len(t) {
        if s[i] == t[j] {
            i++
        }
        j++
    }
    return i == len(s)
}

说明:Golang的代码结构与其他语言类似,但语法和类型声明有所不同。

复杂度分析

  • 时间复杂度:O(m + n),其中 m 和 n 分别是字符串 st 的长度。两个指针至多遍历两个字符串各一次。
  • 空间复杂度:O(1),只使用了常量级别的额外空间。

复杂度分析

  • 时间复杂度:O(m + n),其中 m 和 n 分别是字符串 st 的长度。
  • 空间复杂度:O(k),其中 k 是字符串 t 中不同字符的数量。在最坏情况下,t 中的所有字符都不同,此时空间复杂度为 O(n)。
方式优点缺点时间复杂度空间复杂度
方式一逻辑简单清晰,只需遍历一次字符串无特别显著的缺点O(m + n)O(1)

相似题目

相似题目难度链接
leetcode 115. 不同的子序列困难力扣-115
leetcode 300. 最长递增子序列中等力扣-300
leetcode 53. 最大子序和简单力扣-53

在解决“判断子序列”这类问题时,双指针法是一种非常高效且直观的方法,它能够直接通过比较字符的相对位置关系来判断一个字符串是否是另一个字符串的子序列,而无需使用额外的数据结构或复杂的算法。因此,在实际应用中,我们可以优先考虑使用双指针法来解决类似的问题。

欢迎一键三连(关注+点赞+收藏),技术的路上一起加油!!!代码改变世界

  • 关于我:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名),回复暗号,更能获取学习秘籍和书籍等

  • —⬇️欢迎关注下面的公众号:进朱者赤,认识不一样的技术人。⬇️—

;