首页 > 其他分享 >找最长公共前缀

找最长公共前缀

时间:2022-10-23 22:57:45浏览次数:47  
标签:return 前缀 strs 复杂度 公共 字符串 033 最长 0m

前言:在python中,比较两个字符串的大小,比较的是首个字母的ascll码值,如果首个字母的值相同,就以此类推,继续往下进行比较,直到得出结果为止

1、利用字符串比较大小,找出最长公共前缀:

from typing import List

class Solution:
    def longestCommonPrefix(self, strs: List[str]) -> str:
        if not strs: return ""
        s1 = min(strs)
        s2 = max(strs)
        print(s1, s2)
        for i, x in enumerate(s1):
            if x != s2[i]:
                return s2[:i]
        return s1


s = Solution()
print("\033[32m[+]\033[0m", "\033[33m", s.longestCommonPrefix(["flowers", "flow", "flight", "forjrigdfl"]), "\033[0m")
print("""\033[31m
巧妙利用python比较两个字符串的方法:
    python比较两个字符串比较的是首个字母ascll码值,若相等就比较下一个,以此类推\033[0m
""")

 2、利用横向扫描的方法找出最长公共前缀:

第二种找出公共前缀的方法,定义一个lcp方法用于过滤出两个字符串相同的部分,然后循环调用这个过滤的方法,最后得出这个字符串数组里面的字符串的最长公共前缀:

from typing import List

class Solution1:
    def longestCommonPrefix(self, strs: List[str]) -> str:
        if not strs:
            return ""

        prefix, count = strs[0], len(strs)
        for i in range(1, count):
            prefix = self.lcp(prefix, strs[i])
            if not prefix:
                break
        return prefix

    def lcp(self, str1, str2):
        length, index = min(len(str1), len(str2)), 0
        while index < length and str1[index] == str2[index]:  # 过滤出来相同的部分
            index += 1
        return str1[:index]


s1 = Solution1()
print("\033[32m[+]\033[0m", "\033[33m", s1.longestCommonPrefix(["leecode", "leet", "ledem", "leet"]), "\033[0m")

print("""\033[31m
复杂度分析:
    时间复杂度:O(mn),其中 m 是字符串数组中的字符串的平均长度,n 是字符串的数量。
    最坏情况下,字符串数组中的每个字符串的每个字符都会被比较一次。
    空间复杂度:O(1)。因为使用的额外空间复杂度为常数。\033[0m
""")

3、利用纵向扫描的方法,将字符串的最长前缀找出:

横向扫描是依次遍历每个字符串,更新最长公共前缀。
另一种方法是纵向扫描.纵向扫描时,从前往后遍历所有字符串的每一列,比较相同列上的字符是否相同,
如果相同则继续对下一列进行比较,如果不相同则当前列不再属于公共前缀,当前列之前的部分为最长公共前缀。

from typing import List

class Solution2:
    def longestCommonPrefix(self, strs: List[str]) -> str:
        if not strs:
            return ""

        length, count = len(strs[0]), len(strs)
        for i in range(length):
            c = strs[0][i]
            if any(i == len(strs[j]) or strs[j][i] != c for j in range(1, count)):
                return strs[0][:i]

        return strs[0]


s2 = Solution2()
print('\033[32m[+]\033[0m', "\033[33m", s2.longestCommonPrefix(["dog", "dat", "does", "dpon"]), "\033[0m")
print("""
\033[31m复杂度分析:\033[0m
\033[35m时间复杂度:O(mn),其中 m 是字符串数组中的字符串的平均长度,n 是字符串的数量。最坏情况下,字符串数组中的每个字符串的每个字符都会被比较一次。
        空间复杂度:O(1)。使用的额外空间复杂度为常数。\033[0m
""")

4、可以采用分而治之的方法,先将数组按两个一组进行比较找出最长公共前缀,再以此类推继续进行比较,得到的结果再进行比较,最后合并得到的结果,找出最终的最长公共前缀 

from typing import List

class Solution3:
    def longestCommonPrefix(self, strs: List[str]) -> str:
        def lcp(start, end):
            if start == end:
                return strs[start]

            mid = (start + end) // 2
            lcpLeft, lcpRight = lcp(start, mid), lcp(mid + 1, end)
            minLength = min(len(lcpLeft), len(lcpRight))
            for i in range(minLength):
                if lcpLeft[i] != lcpRight[i]:
                    return lcpLeft[:i]

            return lcpLeft[:minLength]

        return "" if not strs else lcp(0, len(strs) - 1)


s3 = Solution3()
print("\033[45m", s3.longestCommonPrefix(["leetcode", "leet", "leed"]), "\033[0m")
print("""
\033[46m复杂度分析:\033[0m
    \033[45m时间复杂度:O(mn),其中 m 是字符串数组中的字符串的平均长度,n 是字符串的数量。递推式:T(n) = 2*T(n/2) + O(m),计算得T(n) = O(mn)
    空间复杂度:O(m log n),其中 m 是字符串数组中的字符串的平均长度,n 是字符串的数量。空间复杂度主要取决于递归调用的层数,
                层数最大为log n,每层需要 m 的空间存储返回结果。\033[0m
""")

5、二分法

最长公共前缀的长度不会超过字符串数组中的最短字符串的长度。
用 minLength 表示字符串数组中的最短字符串的长度,
则可以在 [0,minLength] 的范围内通过二分查找得到最长公共前缀的长度。
每次取查找范围的中间值 mid,判断每个字符串的长度为mid 的前缀是否相同,
如果相同则最长公共前缀的长度一定大于或等于 mid,如果不相同则最长公共前缀的长度一定小于 mid,
通过上述方式将查找范围缩小一半,直到得到最长公共前缀的长度。

from typing import List

class Solution4:
    def longestCommonPrefix(self, strs: List[str]) -> str:
        def isCommonPrefix(length):
            str0, count = strs[0][:length], len(strs)
            return all(strs[i][:length] == str0 for i in range(1, count))

        if not strs:
            return ""

        minLength = min(len(s) for s in strs)   # 找出最短的字符串
        low, high = 0, minLength
        while low < high:
            mid = (high - low + 1) // 2 + low
            if isCommonPrefix(mid):
                low = mid
            else:
                high = mid - 1

        return strs[0][:low]


s4 = Solution4()
print("\033[45m", s4.longestCommonPrefix(["leetcode", "leet", "leed"]), "\033[0m")
print("""
\033[41m复杂度:\033[0m
    \033[45m时间复杂度:O(mn log m),其中 m 是字符串数组中的字符串的最小长度,
                n 是字符串的数量。二分查找的迭代执行次数是 O(log m),
                每次迭代最多需要比较 mn 个字符,因此总时间复杂度是 O(mn log m)。
    空间复杂度:O(1)。使用的额外空间复杂度为常数。\033[0m

""")

 Finished!

标签:return,前缀,strs,复杂度,公共,字符串,033,最长,0m
From: https://www.cnblogs.com/yncaqy/p/16819911.html

相关文章

  • #yyds干货盘点# LeetCode 腾讯精选练习 50 题:最长公共前缀
    题目:编写一个函数来查找字符串数组中的最长公共前缀。如果不存在公共前缀,返回空字符串 ""。 示例1:输入:strs=["flower","flow","flight"]输出:"fl"示例2:输入:strs=["do......
  • 区间问题----差分+离散化+前缀和
     《二维离散化+二维前缀和+二维双指针算法+二分+以点代二维区间》思路:首先,利用二分,将求解问题变成判断问题,二分边长关于在二维平面上的任意区域的数值问题可......
  • 771. 字符串中最长的连续出现的字符
    文章目录​​Question​​​​Ideas​​​​Code​​Question求一个字符串中最长的连续出现的字符,输出该字符及其出现次数,字符串中无空白字符(空格、回车和tab),如果这样的字......
  • 【算法】算法练习之最长有效括号和两个删除
    算法内容1.最长有效括号,题目类型:栈,字符串,比较困难2.删除有序数组中的重复项,题目类型:数组,双指针,比较简单3.删除有效括号,题目类型:字符串,优先搜索,比较困难题目描述第一......
  • 最长回文子串
    输入:s="babad"输出:"bab"解释:"aba"同样是符合题意的答案。classSolution:deflongestPalindrome(self,s:str)->str:palindrome=""#中心......
  • AcWing 895.最长上升子序列Ⅰ
    题目链接:http://www.acwing.com/problem/content/897/浅浅复习放AC代码1#include<bits/stdc++.h>2usingnamespacestd;34constintN=1010;5intn;......
  • #yyds干货盘点# LeetCode 腾讯精选练习 50 题:最长回文子串
    题目:给你一个字符串s,找到s中最长的回文子串。 示例1:输入:s="babad"输出:"bab"解释:"aba"同样是符合题意的答案。示例2:输入:s="cbbd"输出:"bb"代码实现:publicclassSo......
  • 二叉搜索树的最近公共祖先
    一.递归publicTreeNodelowestCommonAncestor(TreeNoderoot,TreeNodep,TreeNodeq){if(root.val<p.val&&root.val<q.val)return......
  • 799. 最长连续不重复子序列
    给定一个长度为n的整数序列,请找出最长的不包含重复的数的连续区间,输出它的长度。输入格式第一行包含整数n。第二行包含n个整数(均在0∼105范围内),表示整数序列。输出......
  • leetcode 最长回文子串
    constcountSubstrings=(s)=>{conststrLen=s.length;letnumOfPalindromicStr=0;//初始化一个二维数组letdp=Array.from(Array(strLen),()=>A......