每日一题——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

相关文章

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

python 字典d[k]中key不存在的解决方案

《python字典d[k]中key不存在的解决方案》本文主要介绍了在Python中处理字典键不存在时获取默认值的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录defaultdict:处理找不到的键的一个选择特殊方法__missing__有时候为了方便起见,

使用Python绘制可爱的招财猫

《使用Python绘制可爱的招财猫》招财猫,也被称为“幸运猫”,是一种象征财富和好运的吉祥物,经常出现在亚洲文化的商店、餐厅和家庭中,今天,我将带你用Python和matplotlib库从零开始绘制一... 目录1. 为什么选择用 python 绘制?2. 绘图的基本概念3. 实现代码解析3.1 设置绘图画

Python pyinstaller实现图形化打包工具

《Pythonpyinstaller实现图形化打包工具》:本文主要介绍一个使用PythonPYQT5制作的关于pyinstaller打包工具,代替传统的cmd黑窗口模式打包页面,实现更快捷方便的... 目录1.简介2.运行效果3.相关源码1.简介一个使用python PYQT5制作的关于pyinstall

使用Python实现大文件切片上传及断点续传的方法

《使用Python实现大文件切片上传及断点续传的方法》本文介绍了使用Python实现大文件切片上传及断点续传的方法,包括功能模块划分(获取上传文件接口状态、临时文件夹状态信息、切片上传、切片合并)、整... 目录概要整体架构流程技术细节获取上传文件状态接口获取临时文件夹状态信息接口切片上传功能文件合并功能小

python实现自动登录12306自动抢票功能

《python实现自动登录12306自动抢票功能》随着互联网技术的发展,越来越多的人选择通过网络平台购票,特别是在中国,12306作为官方火车票预订平台,承担了巨大的访问量,对于热门线路或者节假日出行... 目录一、遇到的问题?二、改进三、进阶–展望总结一、遇到的问题?1.url-正确的表头:就是首先ur

C#实现文件读写到SQLite数据库

《C#实现文件读写到SQLite数据库》这篇文章主要为大家详细介绍了使用C#将文件读写到SQLite数据库的几种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以参考一下... 目录1. 使用 BLOB 存储文件2. 存储文件路径3. 分块存储文件《文件读写到SQLite数据库China编程的方法》博客中,介绍了文

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

JAVA利用顺序表实现“杨辉三角”的思路及代码示例

《JAVA利用顺序表实现“杨辉三角”的思路及代码示例》杨辉三角形是中国古代数学的杰出研究成果之一,是我国北宋数学家贾宪于1050年首先发现并使用的,:本文主要介绍JAVA利用顺序表实现杨辉三角的思... 目录一:“杨辉三角”题目链接二:题解代码:三:题解思路:总结一:“杨辉三角”题目链接题目链接:点击这里

基于Python实现PDF动画翻页效果的阅读器

《基于Python实现PDF动画翻页效果的阅读器》在这篇博客中,我们将深入分析一个基于wxPython实现的PDF阅读器程序,该程序支持加载PDF文件并显示页面内容,同时支持页面切换动画效果,文中有详... 目录全部代码代码结构初始化 UI 界面加载 PDF 文件显示 PDF 页面页面切换动画运行效果总结主