每日一题——Python实现PAT甲级1077 Kuchiguse(举一反三+思想解读+逐步优化)

本文主要是介绍每日一题——Python实现PAT甲级1077 Kuchiguse(举一反三+思想解读+逐步优化),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


一个认为一切根源都是“自己不够强”的INTJ

个人主页:用哲学编程-CSDN博客
专栏:每日一题——举一反三
Python编程学习
Python内置函数

Python-3.12.0文档解读

目录

我的写法

代码点评

时间复杂度分析

空间复杂度分析

总结

我要更强

方案1:利用字典树(后缀树)

优化代码(后缀树实现)

代码点评

时间复杂度分析

空间复杂度分析

方案2:水平扫描法

优化代码(水平扫描法实现)

代码点评

时间复杂度分析

空间复杂度分析

总结

哲学和编程思想

1. 字典树(Trie)方法

编程思想

哲学理念

2. 水平扫描法

编程思想

哲学理念

3. 动态编程(不在上述代码中,但也是常见的优化方法)

编程思想

哲学理念

总结

举一反三

1. 数据结构优化

2. 空间换时间

3. 递归与迭代

4. 分而治之

5. 贪心算法

6. 动态规划

总结




题目链接

我的写法

N=int(input())  # 读取一个整数N,表示有多少个字符串
strs=[]  # 创建一个空列表来存储字符串
for i in range(N):  # 循环N次strs.append(input()[::-1])  # 读取每个字符串并将其倒序,然后添加到列表strs中same_end=''  # 初始化一个空字符串,用于存储相同的后缀
for i in range(min([len(s) for s in strs])):  # 找出所有字符串中最短的长度,并遍历该长度if len(set([strs[j][i] for j in range(N)]))==1:  # 检查所有字符串在第i个位置上的字符是否相同same_end+=strs[0][i]  # 如果相同,则将该字符添加到same_end中else:  # 如果不相同break  # 结束循环if len(same_end)==0:  # 如果没有相同的后缀print("nai")  # 输出"nai"
else:  # 如果有相同的后缀print(same_end[::-1])  # 将same_end倒序输出,即得到原始的相同后缀

这段代码的功能是找到多个字符串的最长公共后缀。以下是对代码的专业点评以及时间复杂度和空间复杂度分析:

代码点评

  1. 输入处理:
    • 使用 input() 读取输入的整数 N,然后读取 N 个字符串,并将其倒序存储在列表 strs 中。这样做是为了方便从后向前逐字符比较字符串。
  2. 查找公共后缀:
    • 首先计算所有字符串的最小长度,确保不会越界。
    • 逐字符检查所有字符串在当前位是否相同,如果相同则添加到 same_end 中,否则终止检查。
  3. 结果输出:
  • 如果 same_end 为空,则输出 "nai",表示没有公共后缀。
  • 否则将 same_end 倒序输出,即为原始字符串的公共后缀。

时间复杂度分析

因此,查找公共后缀部分的总体时间复杂度为 O(M * N)。

  1. 输入处理:
    • 读取 N 个字符串的时间复杂度是 O(N)。
    • 每个字符串的倒序操作的时间复杂度是 O(L),其中 L 是字符串的平均长度。总体时间复杂度为 O(N * L)。
  2. 查找公共后缀:
    • 计算最小长度的时间复杂度是 O(N)。
    • 逐字符比较的时间复杂度是 O(M * N),其中 M 是最短字符串的长度。
  3. 结果输出:
  • 输出的时间复杂度为 O(M)。

综合以上分析,总的时间复杂度为 O(N * L + M * N),在最坏情况下 L 和 M 都可以接近最长字符串长度。

空间复杂度分析

  1. 输入处理:
    • 使用了一个列表 strs 来存储 N 个字符串,空间复杂度为 O(N * L)。
  2. 查找公共后缀:
  • 使用了一个字符串 same_end 来存储公共后缀,最坏情况的空间复杂度为 O(M)。

综合分析,空间复杂度为 O(N * L + M)。

总结

这段代码在处理字符串公共后缀问题时较为直接,逻辑清晰,步骤明确。代码的时间复杂度和空间复杂度都取决于输入字符串的数量和长度,但对于常规的输入规模,性能表现应该是合理的。优化的重点可以放在减少字符串操作的次数和改进字符串比较过程。


我要更强

在寻找多个字符串的最长公共后缀时,时间复杂度和空间复杂度的优化可以通过多种途径实现,例如避免多余的字符串操作,使用更高效的数据结构等。以下是一些可能的优化方案,并给出相应的代码和注释:

方案1:利用字典树(后缀树)

字典树(Trie)是一种高效的数据结构,特别适合字符串集合的共性处理。在这里,可以使用反向构建的后缀树来查找公共后缀。

优化代码(后缀树实现)

class TrieNode:def __init__(self):self.children = {}self.is_end_of_word = Falseclass Trie:def __init__(self):self.root = TrieNode()def insert(self, word):current = self.rootfor char in word:if char not in current.children:current.children[char] = TrieNode()current = current.children[char]current.is_end_of_word = Truedef longest_common_suffix(self, n):common_suffix = []current = self.rootwhile len(current.children) == 1 and not current.is_end_of_word:(char, next_node), = current.children.items()common_suffix.append(char)current = next_noden -= 1if n == 0:breakreturn ''.join(common_suffix)def find_longest_common_suffix(strings):trie = Trie()for string in strings:trie.insert(string[::-1])return trie.longest_common_suffix(len(strings))N = int(input())  # 读取一个整数N,表示有多少个字符串
strs = [input() for _ in range(N)]  # 读取N个字符串result = find_longest_common_suffix(strs)if not result:print("nai")
else:print(result[::-1])

代码点评

  1. Trie 数据结构:
    • 使用 TrieNode 类表示字典树的节点。
    • 使用 Trie 类封装字典树的插入和寻找最长公共后缀的功能。
  2. 插入字符串:
    • 将每个字符串倒序后插入字典树。
  3. 寻找公共后缀:
  • 在字典树中搜索最长的路径,路径上的字符即为最长公共后缀。

时间复杂度分析

  • 插入每个字符串的时间复杂度为 O(L),其中 L 是字符串的长度,因此总插入时间复杂度为 O(N * L)。
  • 寻找最长公共后缀的时间复杂度为 O(M),其中 M 是最短字符串的长度。

综合时间复杂度为 O(N * L)。

空间复杂度分析

  • 字典树节点的空间复杂度为 O(N * L),因为每个字符都可能需要一个节点。
  • 另外,存储结果的空间复杂度为 O(M)。

综合空间复杂度为 O(N * L)。

方案2:水平扫描法

水平扫描法通过逐一比较字符串,类似于前缀匹配,但是从字符串末尾开始。

优化代码(水平扫描法实现)

def find_longest_common_suffix(strings):if not strings:return ""# 取第一个字符串作为基准suffix = strings[0][::-1]# 逐个比较其他字符串for string in strings[1:]:temp_suffix = ""rev_string = string[::-1]min_length = min(len(suffix), len(rev_string))# 从后向前比较字符for i in range(min_length):if suffix[i] == rev_string[i]:temp_suffix += suffix[i]else:breaksuffix = temp_suffixif not suffix:breakreturn suffixN = int(input())  # 读取一个整数N,表示有多少个字符串
strs = [input() for _ in range(N)]  # 读取N个字符串result = find_longest_common_suffix(strs)if not result:print("nai")
else:print(result[::-1])

代码点评

  1. 水平扫描:
    • 使用第一个字符串作为初始公共后缀的基准。
    • 逐一与后续字符串进行比较,更新公共后缀。
  2. 字符比较:
  • 从后向前逐字符比较,直到找到不同字符或达到最短长度。

时间复杂度分析

  • 比较每个字符串的时间复杂度为 O(M),其中 M 是最短字符串的长度。因此总时间复杂度为 O(N * M)。

空间复杂度分析

  • 存储公共后缀的空间复杂度为 O(M)。

综合空间复杂度为 O(M)。

总结

上述优化方法在时间和空间复杂度上相比原始方案都有一定的提升。字典树(后缀树)方法在处理大量字符串时更高效,而水平扫描法实现简单且易于理解。选择哪种方法可以根据具体的应用场景和性能需求来定。


哲学和编程思想

在优化寻找多个字符串的最长公共后缀的方法时,涉及到一些重要的编程思想和哲学理念。以下是对这些思想的详细说明:

1. 字典树(Trie)方法

编程思想
  • 数据结构优化:字典树是一种特定的树结构,专门用于处理字符串集合,通过将公共前缀合并在一起,可以高效地存储和查询字符串。
  • 空间换时间:字典树在插入和查询时,花费的时间与字符串的长度呈线性关系,因此可以通过增加空间复杂度来降低时间复杂度。
  • 递归与迭代:字典树的构建和查询通常使用递归或迭代的方式来遍历各节点。
哲学理念
  • 分而治之:将问题分解为更小的子问题,通过处理这些子问题来解决整个问题。字典树通过逐层处理每个字符,实现了这一思想。
  • 高效存储与快速访问:字典树在设计上就是为了优化字符串的存储和访问,这反映了在计算机科学中追求效率的哲学理念。

2. 水平扫描法

编程思想
  • 贪心算法:每次选择当前最优的解,在这里,每次比较当前字符串的最长公共后缀并更新。
  • 迭代优化:逐步改进解决方案,通过不断迭代来优化结果。
  • 暴力搜索:尽管是优化的暴力搜索,水平扫描法还是采用了一种直接比较的方式,这是一种简单且直接的思维方式。
哲学理念
  • 最小工作原则:每一步都尽量少做工作,只比较必要的字符。这种方法避免了多余的计算和复杂的逻辑。
  • 逐步求精:通过逐步比较和更新公共后缀,从而逐渐接近最终解决方案。这种逐步优化的思想在哲学上类似于“渐进主义”。

3. 动态编程(不在上述代码中,但也是常见的优化方法)

编程思想
  • 子问题重用:将问题分解为子问题,通过记忆化或表格存储中间结果,避免重复计算。
  • 状态转移:通过状态转移方程来确定从一个状态到另一个状态的关系。
哲学理念
  • 整体与部分:整体问题的解决依赖于部分问题的解决,通过解决部分问题,逐步构建整体解决方案。
  • 平衡与优化:动态规划在时间和空间之间找到平衡,通过空间换时间来优化计算过程。

总结

通过使用不同的数据结构和算法优化,不仅提高了程序的效率,还展示了重要的编程思想和哲学理念。这些方法在计算机科学中都有广泛的应用,体现了在解决问题时对效率、简洁性和可维护性的追求。字典树方法体现了分而治之和高效存储的哲学,而水平扫描法则展示了逐步求精和贪心算法的思想。这些方法的结合,使得可以更好地理解和处理复杂的字符串问题。


举一反三

理解和应用编程思想与哲学理念,可以帮助在遇到各种编程问题时,灵活地找出高效的解决方案。以下是一些技巧,能帮助举一反三,从而更好地应对各种复杂问题:

1. 数据结构优化

技巧:

  • 学习和理解常用的数据结构(如数组、链表、栈、队列、树、图、哈希表等)的特点和适用场景。
  • 在解决问题时,首先考虑是否有合适的数据结构能简化问题或提高效率。
  • 对于特定的应用场景,考虑定制或组合多种数据结构。

举例:

  • 当处理大量字符串时,考虑使用字典树(Trie)来高效地存储和查询公共前缀或后缀。
  • 在频繁需要插入和删除操作时,选择链表而不是数组。

2. 空间换时间

技巧:

  • 当计算时间复杂度较高时,考虑使用额外的空间来存储中间结果,从而减少计算时间。
  • 使用缓存或者记忆化技术(如动态规划中的 memoization)来避免重复计算。

举例:

  • 在动态规划中,使用二维数组存储中间计算结果,优化递归问题的时间复杂度。
  • 在查找问题中,使用哈希表存储已访问节点或结果,优化查找速度。

3. 递归与迭代

技巧:

  • 理解递归和迭代的转换,当递归深度过大时尝试使用迭代来避免栈溢出。
  • 递归适用于分治法,迭代适用于需要多次重复计算的场景。

举例:

  • 使用递归实现快速排序(Quick Sort),但在递归深度可能过大时,改为使用迭代版本。
  • 通过迭代实现斐波那契数列,避免递归重复计算。

4. 分而治之

技巧:

  • 将复杂问题划分为更小的子问题,分别解决这些子问题再合并结果。
  • 使用分治法处理递归问题或大数据量的处理问题。

举例:

  • 使用归并排序(Merge Sort)将大问题拆分成小问题,再合并排序。
  • 在二分搜索中,将问题空间逐步缩小,直到找到目标。

5. 贪心算法

技巧:

  • 每一步选择当前最好(最优)的解,适用于局部最优能导致全局最优的问题。
  • 贪心算法往往简单高效,但需证明所选策略能得到全局最优解。

举例:

  • 在活动选择问题中,每次选择结束时间最早的活动。
  • 在找零问题中,每次选择面值最大的硬币。

6. 动态规划

技巧:

  • 识别问题的子结构,将问题分解为子问题,通过存储子问题的解来避免重复计算。
  • 使用表格法或记忆化技术存储子问题的解,优化计算过程。

举例:

  • 在背包问题中,使用二维数组存储每个容量和物品组合的最优解。
  • 在最长公共子序列问题中,构建二维表格来存储子序列的长度。

总结

通过灵活运用数据结构优化、空间换时间、递归与迭代、分而治之、贪心算法和动态规划等编程思想,可以更高效地解决各种复杂问题。在面对新的问题时,尝试识别其性质,寻找合适的编程思想和数据结构,从而设计出高效的解决方案。多练习和多思考,将这些技巧内化为自己的编程习惯,就能举一反三,游刃有余地解决各种编程挑战。


感谢支持。

这篇关于每日一题——Python实现PAT甲级1077 Kuchiguse(举一反三+思想解读+逐步优化)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1045400

相关文章

使用Python删除Excel中的行列和单元格示例详解

《使用Python删除Excel中的行列和单元格示例详解》在处理Excel数据时,删除不需要的行、列或单元格是一项常见且必要的操作,本文将使用Python脚本实现对Excel表格的高效自动化处理,感兴... 目录开发环境准备使用 python 删除 Excphpel 表格中的行删除特定行删除空白行删除含指定

Linux下删除乱码文件和目录的实现方式

《Linux下删除乱码文件和目录的实现方式》:本文主要介绍Linux下删除乱码文件和目录的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux下删除乱码文件和目录方法1方法2总结Linux下删除乱码文件和目录方法1使用ls -i命令找到文件或目录

SpringBoot+EasyExcel实现自定义复杂样式导入导出

《SpringBoot+EasyExcel实现自定义复杂样式导入导出》这篇文章主要为大家详细介绍了SpringBoot如何结果EasyExcel实现自定义复杂样式导入导出功能,文中的示例代码讲解详细,... 目录安装处理自定义导出复杂场景1、列不固定,动态列2、动态下拉3、自定义锁定行/列,添加密码4、合并

mybatis执行insert返回id实现详解

《mybatis执行insert返回id实现详解》MyBatis插入操作默认返回受影响行数,需通过useGeneratedKeys+keyProperty或selectKey获取主键ID,确保主键为自... 目录 两种方式获取自增 ID:1. ​​useGeneratedKeys+keyProperty(推

Spring Boot集成Druid实现数据源管理与监控的详细步骤

《SpringBoot集成Druid实现数据源管理与监控的详细步骤》本文介绍如何在SpringBoot项目中集成Druid数据库连接池,包括环境搭建、Maven依赖配置、SpringBoot配置文件... 目录1. 引言1.1 环境准备1.2 Druid介绍2. 配置Druid连接池3. 查看Druid监控

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

Linux在线解压jar包的实现方式

《Linux在线解压jar包的实现方式》:本文主要介绍Linux在线解压jar包的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux在线解压jar包解压 jar包的步骤总结Linux在线解压jar包在 Centos 中解压 jar 包可以使用 u

Python办公自动化实战之打造智能邮件发送工具

《Python办公自动化实战之打造智能邮件发送工具》在数字化办公场景中,邮件自动化是提升工作效率的关键技能,本文将演示如何使用Python的smtplib和email库构建一个支持图文混排,多附件,多... 目录前言一、基础配置:搭建邮件发送框架1.1 邮箱服务准备1.2 核心库导入1.3 基础发送函数二、

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

Python包管理工具pip的升级指南

《Python包管理工具pip的升级指南》本文全面探讨Python包管理工具pip的升级策略,从基础升级方法到高级技巧,涵盖不同操作系统环境下的最佳实践,我们将深入分析pip的工作原理,介绍多种升级方... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核