每日一题——8行Python代码实现PAT乙级1029 旧键盘(举一反三+思想解读+逐步优化)五千字好文

news2025/1/11 9:02:16


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

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

Python-3.12.0文档解读

目录

我的写法

代码评析

时间复杂度

空间复杂度

我要更强

方法一:使用集合来存储不正确的字符

方法二:使用列表来存储错误按键

方法三:使用布尔数组来存储不正确的字符

哲学和编程思想

方法一:使用集合来存储不正确的字符

方法二:使用列表来存储错误按键

方法三:使用布尔数组来存储不正确的字符

总结

举一反三

1. 空间换时间

2. 集合论和数据结构选择

3. 算法复杂度分析

4. 简单性和可读性

5. 测试和调试

6. 学习和实践


题目链接

我的写法

# 读取用户输入的字符串,并将其转换为大写
corrects = input().upper()

# 创建一个集合,用于存储错误的按键
bad_keys = set()

# 再次读取用户输入的字符串,并将其转换为大写
incorrects = input().upper()

# 遍历正确字符串的每个字符
for i in range(len(corrects)):
    # 检查当前字符是否存在于不正确的字符串中
    if incorrects.find(corrects[i]) == -1:
        # 如果当前字符不在不正确的字符串中,并且它还没有被添加到错误的按键集合中
        if corrects[i] not in bad_keys:
            # 将该字符添加到错误的按键集合中
            bad_keys.add(corrects[i])
            # 打印出该字符,使用end=''确保字符在同一行输出
            print(corrects[i], end='')

这段代码实现了一个比较简单的功能,即在两个输入字符串中找出那些在第一个字符串中出现但在第二个字符串中没有出现的字符,并且输出这些字符。下面是对这段代码的详细评析,包括时间复杂度和空间复杂度分析。

代码评析

  1. 代码逻辑清晰:代码的逻辑比较直接,通过两次读取输入来获取两个字符串,然后依次检查第一个字符串中的每个字符是否存在于第二个字符串中。这种方法直观且容易理解。
  2. 使用集合提高查找效率:利用集合 bad_keys 来存储已经发现的并输出过的错误字符。这是一个不错的设计,因为集合具有较高的查找效率(O(1) 平均时间复杂度)。
  3. 字符串查找操作:在 incorrects.find(corrects[i]) 中,字符串的 find 方法在最坏情况下的时间复杂度是 O(n),其中 n 是字符串 incorrects 的长度。由于这是在一个循环内进行的(循环次数为 len(corrects)),因此在最坏情况下,这部分代码的时间复杂度是 O(m * n),其中 m 是字符串 corrects 的长度,n 是字符串 incorrects 的长度。

时间复杂度

  • 读取输入:这部分时间复杂度可以忽略不计,因为它依赖于用户输入的速度。
  • 遍历 corrects 字符串:这个循环的时间复杂度是 O(m),其中 m 是 corrects 字符串的长度。
  • 查找字符是否存在于 incorrects 中:使用 find 方法,最坏情况下时间复杂度是 O(n),其中 n 是 incorrects 字符串的长度。这一步在循环内执行,因此总的时间复杂度是 O(m * n)。
  • 集合操作:在最坏情况下,集合的添加和查找操作的时间复杂度是 O(1)。

总结起来,这段代码的整体时间复杂度是 O(m * n),其中 m 是 corrects 字符串的长度,n 是 incorrects 字符串的长度。

空间复杂度

  • 字符串存储:两个输入字符串 corrects 和 incorrects 占用的空间是 O(m + n)。
  • 集合 bad_keys:最坏情况下,集合 bad_keys 中可能包含所有正确字符串中的字符,因此占用的空间是 O(m)。

总结起来,这段代码的空间复杂度是 O(m + n),其中 m 是 corrects 字符串的长度,n 是 incorrects 字符串的长度。


我要更强

当然,有几种优化代码的方式可以同时提高时间复杂度和空间复杂度。我会提供几个方案,并给出完整的代码和注释。

方法一:使用集合来存储不正确的字符

优化思路: 通过将 incorrects 转换为集合,可以利用集合的查找效率(O(1)),从而优化时间复杂度。

代码:

# 读取用户输入的字符串,并将其转换为大写
corrects = input().upper()

# 创建一个集合,用于存储不正确的字符
incorrects = set(input().upper())

# 创建一个集合,用于存储已经检测到的错误按键
bad_keys = set()

# 遍历正确字符串的每个字符
for char in corrects:
    # 检查当前字符是否存在于不正确的字符集合中
    if char not in incorrects:
        # 如果当前字符不在不正确的字符集合中,并且它还没有被添加到错误按键集合中
        if char not in bad_keys:
            # 将该字符添加到错误按键集合中
            bad_keys.add(char)
            # 打印出该字符,使用end=''确保字符在同一行输出
            print(char, end='')

时间复杂度:O(m)

  • 遍历 corrects 字符串的时间复杂度是 O(m)。
  • 在 incorrects 集合中查找字符的时间复杂度是 O(1)。

空间复杂度:O(m + n)

  • 存储 corrects 和 incorrects 共占用 O(m + n) 的空间。
  • 存储 bad_keys 的空间是 O(k),其中 k 是错误按键的数量,最坏情况下 k = m。

方法二:使用列表来存储错误按键

优化思路: 列表的空间效率高于集合,如果错误按键数量较少,这种方法可以节省一些空间。

代码:

# 读取用户输入的字符串,并将其转换为大写
corrects = input().upper()

# 创建一个集合,用于存储不正确的字符
incorrects = set(input().upper())

# 创建一个列表,用于存储已经检测到的错误按键
bad_keys = []

# 遍历正确字符串的每个字符
for char in corrects:
    # 检查当前字符是否存在于不正确的字符集合中
    if char not in incorrects:
        # 如果当前字符不在不正确的字符集合中,并且它还没有被添加到错误按键列表中
        if char not in bad_keys:
            # 将该字符添加到错误按键列表中
            bad_keys.append(char)
            # 打印出该字符,使用end=''确保字符在同一行输出
            print(char, end='')

时间复杂度:O(m)

  • 遍历 corrects 字符串的时间复杂度是 O(m)。
  • 在 incorrects 集合中查找字符的时间复杂度是 O(1)。
  • 在列表 bad_keys 中查找字符的时间复杂度是 O(k),最坏情况下 k = m。

空间复杂度:O(m + n)

  • 存储 corrects 和 incorrects 共占用 O(m + n) 的空间。
  • 存储 bad_keys 的空间是 O(k),其中 k 是错误按键的数量。

方法三:使用布尔数组来存储不正确的字符

优化思路: 如果字符集的范围有限(如仅字母A-Z),可以使用固定大小的布尔数组替代集合,这样可以进一步优化空间和查找效率。

代码:

# 读取用户输入的字符串,并将其转换为大写
corrects = input().upper()

# 创建一个布尔数组,用于标记不正确的字符
incorrects = [False] * 26

# 填充不正确的字符布尔数组
for char in input().upper():
    incorrects[ord(char) - ord('A')] = True

# 创建一个集合,用于存储已经检测到的错误按键
bad_keys = set()

# 遍历正确字符串的每个字符
for char in corrects:
    # 检查当前字符是否存在于不正确的字符布尔数组中
    if not incorrects[ord(char) - ord('A')]:
        # 如果当前字符不在不正确的字符布尔数组中,并且它还没有被添加到错误按键集合中
        if char not in bad_keys:
            # 将该字符添加到错误按键集合中
            bad_keys.add(char)
            # 打印出该字符,使用end=''确保字符在同一行输出
            print(char, end='')

时间复杂度:O(m)

  • 遍历 corrects 字符串的时间复杂度是 O(m)。
  • 在布尔数组中查找字符的时间复杂度是 O(1)。

空间复杂度:O(m + 26)

  • 存储 corrects 需要 O(m) 的空间,布尔数组固定为 26 个元素。
  • 存储 bad_keys 的空间是 O(k),其中 k 是错误按键的数量,最坏情况下 k = m。

通过这几种方法,可以有效地优化代码的时间复杂度和空间复杂度,具体选择哪种方法可以根据问题的具体场景和输入数据的特点来决定。


哲学和编程思想

这些方法背后应用了一些重要的编程哲学和思想,如空间换时间、集合论、算法复杂度分析和数据结构选择。下面具体说明这些方法中使用的编程哲学和思想:

方法一:使用集合来存储不正确的字符

编程思想:

  1. 空间换时间:
    • 通过使用集合来存储不正确的字符,牺牲了一部分空间来换取查找操作的高效性(O(1))。这是一个典型的“空间换时间”的策略,用更多的存储空间来减少时间复杂度。
  2. 集合论:
    • 集合的数据结构利用哈希表实现了元素的快速查找、添加和删除操作。这种高效的集合操作源于集合论中的基本概念。
  3. 数据结构选择:
  • 针对查找频繁且不需要保持顺序的情况,选择集合这种数据结构是非常合适的。集合能快速判断某个元素是否存在,非常适用于这类问题。

方法二:使用列表来存储错误按键

编程思想:

  1. 简单性和可读性:
    • 使用列表来存储已经检测到的错误按键,这种方法更简单和直观。但在最坏情况下会影响性能(在列表中的查找操作是O(k))。
  2. 时间复杂度分析:
  • 充分理解算法的时间复杂度,通过对算法的时间复杂度进行分析,决定使用合适的数据结构。例如,在这里为了简化代码,选择列表,但要注意最坏情况下的性能影响。

方法三:使用布尔数组来存储不正确的字符

编程思想:

  1. 空间换时间:
    • 这一方法再一次利用了“空间换时间”的哲学,通过一个固定大小的布尔数组,用更大的空间来换取查找操作的高效性(O(1))。
  2. 数组的使用:
    • 在字符集范围有限的情况下,使用数组(布尔数组)来优化性能。这是通过对问题的特性进行分析后做出的数据结构选择。
  3. 算法复杂度分析:
  • 充分理解和分析算法的时间和空间复杂度,针对问题的特点选择最优的数据结构和算法。布尔数组在字符集范围有限的情况下,可以极大地提高查找操作的效率。

总结

这些方法背后都体现了一些关键的编程哲学和思想:

  1. 空间换时间:
    • 通过增加空间使用来换取更快的时间效率,这是高效算法设计中的常见策略。
  2. 集合论和数据结构选择:
    • 根据具体需求选择合适的数据结构(如集合、列表、数组等),利用其特性来优化算法。集合适合频繁查找操作,数组适合空间固定且范围有限的情况。
  3. 算法复杂度分析:
    • 对时间复杂度和空间复杂度进行分析,选择最优的算法和数据结构。在设计和优化算法时,这种分析是至关重要的。
  4. 简单性和可读性:
  • 在保证效率的前提下,代码的简单性和可读性也是重要的考虑因素。简单易懂的代码更容易维护和调试。

这些编程哲学和思想不仅仅适用于这三个方法,对于很多算法和数据结构的设计与优化都具有指导意义。


举一反三

根据上述提到的编程哲学和思想,以下是一些技巧和策略,可以帮助在面对不同编程问题时举一反三:

1. 空间换时间

  • 缓存:对于计算量大的函数或方法,使用缓存(如字典或集合)存储已经计算过的结果,避免重复计算。
  • 预处理:对于需要频繁查询的数据,可以预先处理数据,如排序、建立索引等,以加快后续的查询速度。

2. 集合论和数据结构选择

  • 选择合适的数据结构:根据操作的类型和频率选择最合适的数据结构。例如,如果需要频繁查找元素,使用集合或字典;如果需要保持元素的插入顺序,使用列表或链表。
  • 利用数据结构特性:了解不同数据结构的特性,如栈的“后进先出”,队列的“先进先出”,以及树和图的遍历方式,可以帮助设计更高效的算法。

3. 算法复杂度分析

  • 时间复杂度分析:在编写代码前,先分析算法的时间复杂度,确保算法在可接受的时间范围内运行。
  • 空间复杂度分析:同样地,分析算法的空间复杂度,确保不会因为内存使用过多而导致程序崩溃或效率低下。
  • 优化算法:通过减少循环嵌套、避免不必要的计算等方法来优化算法的时间复杂度。

4. 简单性和可读性

  • 代码重构:定期重构代码,使其更加简洁、清晰,易于理解和维护。
  • 使用有意义的变量和函数名:使用描述性的变量名和函数名,使代码自文档化,减少注释的需求。
  • 模块化编程:将代码分解为小的、独立的模块或函数,每个模块负责一个明确的功能,便于管理和复用。

5. 测试和调试

  • 单元测试:为代码编写单元测试,确保每个模块的功能正确。
  • 调试技巧:学会使用调试工具,如断点、单步执行等,快速定位和修复错误。

6. 学习和实践

  • 持续学习:不断学习新的编程语言、框架和工具,保持技术更新。
  • 实践:通过实际项目或编程练习来应用所学知识,实践是提高编程能力的最佳方式。

通过掌握这些技巧和策略,将能够更加灵活地应对各种编程挑战,提高解决问题的效率和质量。记住,编程不仅仅是写代码,更是一种解决问题的思维方式。通过不断学习和实践,你将能够更好地理解和应用这些编程哲学和思想。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1834088.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

OpenGL Super Bible 7th - Drawing Our First Triangle(绘制第一个三角形)

简介 本文的原版为《OpenGL Super Bible 7th》,是同事给我的,翻译是原文+译文的形势。文章不属于机器直译,原因在于语言不存在一一对应的关系,我将尽可能的按照中国人看起来舒服的方式来翻译这些段子,如果段子让你感到身心愉悦,那还劳烦点个关注,追个更。如果我没有及时…

IT入门知识第四部分《数据库》(4/10)

目录 1. 数据库基础 1.1 数据库的定义 1.2 数据库的关键概念 数据模型 数据库架构 数据库操作语言(DML 和 DDL) 总结 2. 关系型数据库 2.1 MySQL MySQL 的历史和特点 MySQL 的安装和配置 MySQL 的基本操作 2.2 PostgreSQL PostgreSQL 的特…

软件测试期末复习

软件测试期末复习 Author 雨 2024年6月18日 1. 什么是软件测试 从一个通常为无限的执行域中选取合适的有限的测试用例,对程序所期望的行为进行动态验证的活动过程。 2. 软件测试的目的 尽早地发现软件的缺陷 3.什么是测试什么是缺陷 从软件内部看:软件开…

在 Stable Diffusion 中控制光线的三种方式

光线在摄影中扮演着至关重要的角色,并对图像的整体质量和意境产生重要影响。你可以利用光线来增强主题,创造深度和立体感,传达情感,并突出重要细节。 在本文中,你将了解通过以下方法来控制光线: 光线提示…

【推荐100个unity插件之21】unity实现多语言切换功能——Localization插件的使用

文章目录 前言优缺点优点缺点 安装创建配置选择语言选择默认语言创建多语言表数据创建key配置不同语言文本预加载绑定不同多语言文本数据,并显示语言切换自己编写按钮控制语言切换多语言图片切换在构建中使用Localization分析错误修复动态修改多语言文本内容参考推荐…

高考后的抉择:如何在心仪专业与知名学校之间做出选择?

目录 前言1. 专业选择的深度探讨1.1 专业的优势与挑战1.1.1 课程学习1.1.2 就业前景 1.2 专业选择的个人经验与思考 2. 名校对个人发展的长短期影响2.1 名校声誉的品牌效应2.1.1 职业发展2.1.2 社会认可度 2.2 教育资源与学术氛围2.2.1 教育资源2.2.2 学术氛围 2.3 就业优势 3.…

【2024最新精简版】RabbitMQ面试篇

文章目录 Kafka和RabbitMQ什么区别惰性队列(Lazy Queues)是怎么实现的?RabbitMQ工作模式有哪些 ?你们项目中哪里用到了RabbitMQ ?为什么会选择使用RabbitMQ ? 有什么好处 ?使用RabbitMQ如何保证消息不丢失 ?消息的重复消费问题如何解决的…

3.1、前端异步编程(超详细手写实现Promise;实现all、race、allSettled、any;async/await的使用)

前端异步编程规范 Promise介绍手写Promise(resolve,reject)手写Promise(then)Promise相关 API实现allraceallSettledany async/await和Promise的关系async/await的使用 Promise介绍 Promise是一个类,可以翻…

Spring技术——介绍、初识

从这篇博客正式开始学习 Spring 。 一、整个Spring技术学习的介绍 首先先对 Spring 做一个简单的认识,并从以下3个方面了解学习Spring 技术的意义: (1)为什么要学习 Spring 技术? 1、它的市场的占有率很高&#xff…

webpack 压缩图片

压缩前: 压缩后: 压缩后基本上是压缩了70-80%左右 1.依赖版本及配置 "imagemin-webpack-plugin": "^2.4.2", "imagemin-mozjpeg": "^7.0.0", "imagemin-pngquant": "^5.0.1", "webpa…

ClickHouse安装与下载22.3.2.2

ClickHouse安装与下载 目录 1. ClickHouse简介 1.1 ClickHouse优点: 1.2 ClickHouse缺点: 1.3 ClickHouse引擎: 1.3.1 数据库引擎 1.3.2 表引擎 2. ClickHouse下载安装 2.1 ClickHouse下载安装 2.2 ClickHouse使用 1. ClickHouse简…

五要素便携式气象站

TH-BQX5自然灾害,如暴雨、洪水、飓风等,对于人类生活的影响和威胁始终不可忽视。在应对自然灾害的过程中,提高应急响应和预警能力成为了一项迫切而重要的任务。近年来,五要素便携式气象站因其便捷、精准的特点,在自然灾…

大数据技术与实践学习

大数据技术与实践期末复习 生态系统 每一张图片都值得思考,理清楚到底是什么! 1.结构化数据(数据库里面的):Sqoop(效率比较慢/隔一段时间抽取一次) 2.半结构化或者非结构化数据:F…

Coursera耶鲁大学金融课程:Financial Markets 笔记Week 02

Financial Markets 本文是学习 https://www.coursera.org/learn/financial-markets-global这门课的学习笔记 这门课的老师是耶鲁大学的Robert Shiller https://en.wikipedia.org/wiki/Robert_J._Shiller Robert James Shiller (born March 29, 1946)[4] is an American econom…

细说MCU通过定时器输出PWM波形的实现方法

目录 一、硬件及工程 二、建立工程 三、代码修改 1、重定义回调函数 2、启动定时器中断 四、下载运行 一、硬件及工程 文章依赖的硬件及工程配置参考本文作者的其他文章:细说ARM MCU的串口接收数据的实现过程-CSDN博客 细说ARM MCU的串口接收数据的实现过程-C…

Matlab基础语法:变量和数据类型,基本运算,矩阵和向量,常用函数,脚本文件

目录 一、变量和数据类型 二、基本运算 三、矩阵和向量 四、常用函数 五、脚本文件 六、总结 一、变量和数据类型 Matlab 支持多种数据类型,包括数值类型、字符类型和逻辑类型。掌握这些基本的变量和数据类型,是我们进行数学建模和计算的基础。 数…

IT入门知识第三部分《软件开发》(3/10)

目录 IT入门知识大纲第三部分《软件开发》 1. 软件开发生命周期(SDLC) 1.1 需求分析 1.2 软件设计 1.3 程序编码 1.4 软件测试 1.5 项目部署 1.6 运行维护 2. 软件开发方法论 2.1 瀑布模型 2.2 敏捷开发 2.2.1 Scrum 2.2.2 Kanban 2.3 Dev…

ip地址b类和c类的区别是什么?体现在哪里

在数字化时代的浪潮中,IP地址扮演着不可或缺的角色。它们不仅是设备在网络中的唯一标识,更是网络连接和信息传输的基石。然而,对于非专业人士来说,IP地址的复杂分类可能会显得有些晦涩难懂。今天,我们就来深入探索IP地…

C#语言中的Async/await最佳实践

自从 C# 5 中引入 async/await 以来,开发人员之间一直对 async/await 关键字的最佳实践以及幕后实际发生的事情感到困惑。 让我们先从基础开始。 在 Windows 窗体的早期,UI 延迟与 I/O 操作所花费的时间成正比。这意味着,如果您尝试将数据保…

【ONE·基础算法 || 记忆化搜索】

总言 主要内容:编程题举例,熟悉理解记忆化搜索类题型(对比递归、动态规划理解运用)。             文章目录 总言1、记忆化搜索1.1、基本介绍1.2、细节理解(记忆搜索化、递归、动态规划……) 2、斐…