热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

【数组和字符串】(三)字符串简介

目录三、字符串简介3.1最长公共前缀3.2最长回文子串3.3翻转字符串里的单词3.4实现strStr(选修字符串匹配算法:KMP)(未完)三、字符串简介3

目录

三、字符串简介

3.1 最长公共前缀

3.2 最长回文子串

3.3 翻转字符串里的单词

3.4 实现 strStr (选修字符串匹配算法:KMP) (未完)



三、字符串简介

3.1 最长公共前缀


3.1.1 问题描述




3.1.2 求解过程

法一:注意是“公共”前缀,而 不是“出现次数最多的”前缀,因此,每次只需要判断前缀切片出现次数是否等于列表中字符串总数即可。一旦前缀切片出现频次小于字符串总数,则不满足公共,停止寻找。(审题很重要!)

2020/06/04 - 24.13% - 使用了太多技巧,复杂度太高!效率很低!且冗余计算很多!

class Solution:def longestCommonPrefix(self, strs: List[str]) -> str:num = len(strs) # 字符串总数if num == 0: # 特殊情况 []return ""if num == 1: # 特殊情况 [""] ["ab"]return strs[0]f = "" # 初始化输出前缀字符串max_len = len(max(strs, key=lambda x: len(x))) # 最长字符串长度for i in range(1, max_len+1): # 使用切片不会有 index out of range 的问题f_temp, n_temp = collections.Counter(map(lambda y: y[:i], strs)).most_common()[0] if n_temp == num: # 如果该前缀出现频次=字符串总数f = f_temp # 将其作为输出前缀字符串else:return freturn f

法二:注意到,作为“公共”前缀,事实上根本不需要那么多技巧和遍历。只需要选其中一个单词切片,遍历数组依次对比以确定当前切片是否是“公共”的。一旦发现非公共,直接返回当前切片。否则,扩大切片范围再遍历一次。

2020/06/04 - 77.66% - 思路有对,可以继续优化!

class Solution:def longestCommonPrefix(self, strs: List[str]) -> str:if len(strs) == 0:return ""result = "" # 初始化输出结果first_str = strs[0] # 首单词for i in range(1, len(first_str)+1): # 根据参考答案, 使用 find 可以简化 for 后操作cur_front = first_str[:i] # 当前前缀#for j in range(1, length): # 用单纯的 for 比 enumerate 慢 20%# if strs[j][:i] != cur_front:for _, cur_str in enumerate(strs):if cur_str[:i] != front: # 但凡发现一个不一样, 直接 returnreturn resultresult = cur_front # 遍历数组, 确定公共前缀return result

法三:参考写法,太强了,充分利用 Python 特性:zip() 拉链函数 + * 打包 + 集合 set 等实现最优化,化繁为简,佩服!经过观察,最快的方法 无一不是使用 zip() 函数

2020/06/04 - 99.99% - 最快!

class Solution:def longestCommonPrefix(self, strs: List[str]) -> str:res = ""# 通过 zip 缝合所有元素, 超过最短的部分会被截断for i in zip(*strs): #print(i)if len(set(i)) == 1: # 使用集合判断当前位置字符是否公共res += i[0]else:break return res

3.2 最长回文子串


3.2.1 问题描述

3.2.2 求解过程

法一:朴素法,从宽到窄的滑动窗口,依次对其中的字符串切片进行正、逆序比较。一旦符合比较,就是当前最长的回文字符串了,直接返回之。否则,找到最后也没有,就随意返回一个字符。复杂度 O(n^3)。

2020/06/04 - 36.83% - 看来不怎么样!(4752 ms太低效了)

class Solution:def longestPalindrome(self, s: str) -> str:length &#61; len(s)if length <&#61; 1: # 特殊情况如 "" "a"return stimes &#61; 1 # 首次用全长窗口, 只需要比较1次for width in range(length, 0, -1): # 滑动窗口宽度-1for start in range(times): # 滑动窗口移动次数 timeswindow &#61; s[start:start&#43;width] # 滑动窗口内字符串切片if window &#61;&#61; window[::-1]: # 正序逆序比较return windowtimes &#43;&#61; 1 # 滑动窗口宽度-1时,移动次数&#43;1return s[0] # 如果一直找不到就随意返回一个字符

法二&#xff1a;参考答案&#xff0c;原理待回顾

2020/06/04 - 99.94% - 最快&#xff01;(52 ms)

class Solution:def longestPalindrome(self, s: str) -> str:length &#61; len(s)if length <2 or s &#61;&#61; s[::-1]:return selse:max_length &#61; 1 start_pointer &#61; 0 for i in range(1, length):even &#61; s[i-max_length: i&#43;1] # 切片odd &#61; s[i-max_length-1: i&#43;1] # 切片if (i - max_length-1 >&#61; 0) and (odd &#61;&#61; odd[::-1]):start_pointer &#61; i - max_length - 1 max_length &#43;&#61; 2continueif (i - max_length >&#61; 0) and (even &#61;&#61; even[::-1]):start_pointer &#61; i - max_lengthmax_length &#43;&#61; 1return s[start_pointer: start_pointer&#43;max_length]

3.3 翻转字符串里的单词


3.3.1 问题描述

 

3.3.2 求解过程

法一&#xff1a;朴素法&#xff0c;使用 split(" ") 将原字符串以 " " 为间隔划分为字符串列表&#xff0c;使用 [::-1] 反转字符串列表。注意&#xff0c;此时字符串列表仍包含有空字符串。为此&#xff0c;通过 for 循环筛选出非空字符串并加入结果字符串 res 中&#xff0c;然后加入字符串间隔空格。最后&#xff0c;使用 str.rstrip() 删除末端空格。复杂度 O(n)。

2020/06/06 - 90.57% - 朴素的思路

class Solution:def reverseWords(self, s: str) -> str:res &#61; ""tmp &#61; s.split(" ")[::-1]for word in tmp:if word:res &#43;&#61; wordres &#43;&#61; " "return res.rstrip()

法二&#xff1a;组合 Python 字符串方法 split() 和 join() 实现拆分和重组&#xff0c;通过 list[::-1] 反转中间形式的字符串列表。复杂度 O(n)。

2020/06/06 - 99.99% - 虽然最快&#xff0c;但是太依赖内建方法。

class Solution:def reverseWords(self, s: str) -> str:if not s:return ""word_list &#61; s.split()[::-1] # 获取所有有效的字符串列表并反转strs &#61; " ".join(word_list) # 将列表元素以" "为间隔合成新字符串return strs# return " ".join(s.split()[::-1]) # 以上可一言以蔽之

3.4 实现 strStr (选修字符串匹配算法&#xff1a;KMP)

3.4.1 问题描述

3.4.2 求解过程

法一&#xff1a;(基准) 直接使用 Python 字符串内置方法 str.find() 实现查找匹配。如果这样写&#xff0c;面试官肯定要把你踢出去的......

2020/06/07 - 60.60% - 看来还不是最快的&#xff01;

class Solution:def strStr(self, haystack: str, needle: str) -> int:return haystack.find(needle)

 法二&#xff1a;(暴力匹配改) 不同于每次都在 haystack 上滑动窗口取出切片来与 needle 比较的暴力匹配方式&#xff0c;本方法先试探性比较首字母&#xff0c;倘若相同才有兴趣比较全部&#xff0c;从而节省了大量时间。复杂度 O(n) 吗&#xff1f;(怎么算啊&#xff1f;)

202/06/08 - 97.06% - 看来还行

class Solution:def strStr(self, haystack: str, needle: str) -> int:if not needle: # len(needle) &#61;&#61; 0 返回 0return 0lenh &#61; len(haystack)lenn &#61; len(needle)if lenh

法三&#xff1a;(KMP 算法)&#xff1a;待续。。。


推荐阅读
  • 模板引擎StringTemplate的使用方法和特点
    本文介绍了模板引擎StringTemplate的使用方法和特点,包括强制Model和View的分离、Lazy-Evaluation、Recursive enable等。同时,还介绍了StringTemplate语法中的属性和普通字符的使用方法,并提供了向模板填充属性的示例代码。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 本文介绍了计算机网络的定义和通信流程,包括客户端编译文件、二进制转换、三层路由设备等。同时,还介绍了计算机网络中常用的关键词,如MAC地址和IP地址。 ... [详细]
  • 这篇文章主要介绍了Python拼接字符串的七种方式,包括使用%、format()、join()、f-string等方法。每种方法都有其特点和限制,通过本文的介绍可以帮助读者更好地理解和运用字符串拼接的技巧。 ... [详细]
  • JavaScript和HTML之间的交互是经由过程事宜完成的。事宜:文档或浏览器窗口中发作的一些特定的交互霎时。能够运用侦听器(或处置惩罚递次来预订事宜),以便事宜发作时实行相应的 ... [详细]
  • 超级简单加解密工具的方案和功能
    本文介绍了一个超级简单的加解密工具的方案和功能。该工具可以读取文件头,并根据特定长度进行加密,加密后将加密部分写入源文件。同时,该工具也支持解密操作。加密和解密过程是可逆的。本文还提到了一些相关的功能和使用方法,并给出了Python代码示例。 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 学习SLAM的女生,很酷
    本文介绍了学习SLAM的女生的故事,她们选择SLAM作为研究方向,面临各种学习挑战,但坚持不懈,最终获得成功。文章鼓励未来想走科研道路的女生勇敢追求自己的梦想,同时提到了一位正在英国攻读硕士学位的女生与SLAM结缘的经历。 ... [详细]
  • Python实现变声器功能(萝莉音御姐音)的方法及步骤
    本文介绍了使用Python实现变声器功能(萝莉音御姐音)的方法及步骤。首先登录百度AL开发平台,选择语音合成,创建应用并填写应用信息,获取Appid、API Key和Secret Key。然后安装pythonsdk,可以通过pip install baidu-aip或python setup.py install进行安装。最后,书写代码实现变声器功能,使用AipSpeech库进行语音合成,可以设置音量等参数。 ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • 无损压缩算法专题——LZSS算法实现
    本文介绍了基于无损压缩算法专题的LZSS算法实现。通过Python和C两种语言的代码实现了对任意文件的压缩和解压功能。详细介绍了LZSS算法的原理和实现过程,以及代码中的注释。 ... [详细]
  • 本文介绍了在处理不规则数据时如何使用Python自动提取文本中的时间日期,包括使用dateutil.parser模块统一日期字符串格式和使用datefinder模块提取日期。同时,还介绍了一段使用正则表达式的代码,可以支持中文日期和一些特殊的时间识别,例如'2012年12月12日'、'3小时前'、'在2012/12/13哈哈'等。 ... [详细]
  • HashMap的相关问题及其底层数据结构和操作流程
    本文介绍了关于HashMap的相关问题,包括其底层数据结构、JDK1.7和JDK1.8的差异、红黑树的使用、扩容和树化的条件、退化为链表的情况、索引的计算方法、hashcode和hash()方法的作用、数组容量的选择、Put方法的流程以及并发问题下的操作。文章还提到了扩容死链和数据错乱的问题,并探讨了key的设计要求。对于对Java面试中的HashMap问题感兴趣的读者,本文将为您提供一些有用的技术和经验。 ... [详细]
  • MySQL多表数据库操作方法及子查询详解
    本文详细介绍了MySQL数据库的多表操作方法,包括增删改和单表查询,同时还解释了子查询的概念和用法。文章通过示例和步骤说明了如何进行数据的插入、删除和更新操作,以及如何执行单表查询和使用聚合函数进行统计。对于需要对MySQL数据库进行操作的读者来说,本文是一个非常实用的参考资料。 ... [详细]
author-avatar
mobiledu2502870193
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有