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

相关文章

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

uniapp接入微信小程序原生代码配置方案(优化版)

uniapp项目需要把微信小程序原生语法的功能代码嵌套过来,无需把原生代码转换为uniapp,可以配置拷贝的方式集成过来 1、拷贝代码包到src目录 2、vue.config.js中配置原生代码包直接拷贝到编译目录中 3、pages.json中配置分包目录,原生入口组件的路径 4、manifest.json中配置分包,使用原生组件 5、需要把原生代码包里的页面修改成组件的方

16.Spring前世今生与Spring编程思想

1.1.课程目标 1、通过对本章内容的学习,可以掌握Spring的基本架构及各子模块之间的依赖关系。 2、 了解Spring的发展历史,启发思维。 3、 对 Spring形成一个整体的认识,为之后的深入学习做铺垫。 4、 通过对本章内容的学习,可以了解Spring版本升级的规律,从而应用到自己的系统升级版本命名。 5、Spring编程思想总结。 1.2.内容定位 Spring使用经验

通过SSH隧道实现通过远程服务器上外网

搭建隧道 autossh -M 0 -f -D 1080 -C -N user1@remotehost##验证隧道是否生效,查看1080端口是否启动netstat -tuln | grep 1080## 测试ssh 隧道是否生效curl -x socks5h://127.0.0.1:1080 -I http://www.github.com 将autossh 设置为服务,隧道开机启动

Python 字符串占位

在Python中,可以使用字符串的格式化方法来实现字符串的占位。常见的方法有百分号操作符 % 以及 str.format() 方法 百分号操作符 % name = "张三"age = 20message = "我叫%s,今年%d岁。" % (name, age)print(message) # 我叫张三,今年20岁。 str.format() 方法 name = "张三"age

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

vue项目集成CanvasEditor实现Word在线编辑器

CanvasEditor实现Word在线编辑器 官网文档:https://hufe.club/canvas-editor-docs/guide/schema.html 源码地址:https://github.com/Hufe921/canvas-editor 前提声明: 由于CanvasEditor目前不支持vue、react 等框架开箱即用版,所以需要我们去Git下载源码,拿到其中两个主

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

基于Springboot + vue 的抗疫物质管理系统的设计与实现

目录 📚 前言 📑摘要 📑系统流程 📚 系统架构设计 📚 数据库设计 📚 系统功能的具体实现    💬 系统登录注册 系统登录 登录界面   用户添加  💬 抗疫列表展示模块     区域信息管理 添加物资详情 抗疫物资列表展示 抗疫物资申请 抗疫物资审核 ✒️ 源码实现 💖 源码获取 😁 联系方式 📚 前言 📑博客主页:

探索蓝牙协议的奥秘:用ESP32实现高质量蓝牙音频传输

蓝牙(Bluetooth)是一种短距离无线通信技术,广泛应用于各种电子设备之间的数据传输。自1994年由爱立信公司首次提出以来,蓝牙技术已经经历了多个版本的更新和改进。本文将详细介绍蓝牙协议,并通过一个具体的项目——使用ESP32实现蓝牙音频传输,来展示蓝牙协议的实际应用及其优点。 蓝牙协议概述 蓝牙协议栈 蓝牙协议栈是蓝牙技术的核心,定义了蓝牙设备之间如何进行通信。蓝牙协议