我们来看看三种Python反转字符串方法的性能差距

简介: 我们来看看三种Python反转字符串方法的性能差距

阅读本文需要5.2分钟

  反转Python字符串的三种主要方法:“切片”,反转迭代和经典的就地反转算法。


在Python中反转字符串的最佳方法是什么?当然,在日常编程中并不经常使用字符串反转,但是这是一个受欢迎的面试问题:


#你有这个:
'TURBO'
#而您想要的是:
'OBRUT'

这个问题的一种变化是编写一个函数,该函数检查给定的字符串是否是回文,


def is_palindrome(string):
    reversed_string = # ???
    return string == reversed_string
>>> is_palindrome('TACOCAT')
True
>>> is_palindrome('TURBO')
False

显然,我们需要弄清楚如何反转字符串以is_palindrome在Python中实现此功能应该怎么做?

Pythonstr字符串对象没有内置.reverse()方法,就像其他语言(例如Java或C#)进入Python时所呈现的那样,以下方法将会报错

    >>> 'TURBO'.reverse()
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    AttributeError: 'str' object has no attribute 'reverse'
    

    这次我们将介绍在Python中反转字符串的三种主要方法以及比较三者之间的性能差距。


    第一种:使用“ [::-1]切片技巧反转Python字符串

    字符串遵循Python中的序列协议。并且所有序列都支持一个强大的功能,称为切片。您可以将切片视为方括号索引语法的扩展。

    它包括一个特殊情况,其中用“ [::-1]切片序列会产生反向副本。因为Python字符串是序列,所以这是获取字符串的反向副本的快速简便的方法:



    >>>  'TURBO' [:: - 1 ]
    'OBRUT'

    可以将此切片表达式写到一个函数中,让代码的作用更加明显:


    defreverse_string1(s):
        """Return a reversed copyof `s`"""
        returns[::-1]
    >>>reverse_string1('TURBO')
    'OBRUT'

    新手第一次遇到列表切片时可能很难理解列表切片。

    我觉得使用Python的切片功能来反转字符串是一个不错的解决方案,但是对于初学者来说可能很难理解。

    继续…



    第二种:使用reversed()和反转Python字符串str.join()


    使用reverse()内置的reverse迭代来反转字符串。从而得到一个反向迭代器,然后循环遍历字符串中的元素。例如:


      >>>foreleminreversed('TURBO'):
      ...     print(elem)
      O
      B
      R
      U
      T

      使用reversed()不会修改原始字符串(由于Python中的字符串是不可变的,因此不会起作用。)

      到目前为止,所看到的只是如何以相反的顺序遍历字符串的字符。但是,如何使用reverse()函数使用这种方法创建Python字符串的反向副本呢?

      看以下例子


      >>> ''.join(reversed('TURBO'))
      'OBRUT'

      此代码段使用该.join()方法将反向迭代产生的所有字符合并到一个新字符串中。

      当然,还可以再次将此代码写到单独的函数中创建适当的“反向字符串”。例如:


      defreverse_string2(s):
          """Return a reversed copyof `s`"""
          return"".join(reversed(s))
      >>>reverse_string2('TURBO')
      'OBRUT'

      它可以清楚地表达正在发生事情的过程,即使是小白也可以直观地了解到正在执行的过程。



      第三种:移植到Python的“经典”就地字符串反转算法


      这是移植到Python的“经典”就地字符串反转算法。因为Python字符串是不可变的,所以首先需要将输入字符串转换为可变的字符列表,就可以执行就地字符交换:


      defreverse_string3(s):
          """Return a reversed copyof `s`"""
          chars=list(s)
          foriinrange(len(s)//2):
              tmp=chars[i]
              chars[i]=chars[len(s)-i-1]
              chars[len(s)-i-1]=tmp
          return''.join(chars)
      >>>reverse_string3('TURBO')
      'OBRUT'

      但是,此方法方非常不实用,它没有发挥Python的优势,并且基本上是C算法的直接移植。哈哈哈,估计大家都不考虑吧

      接下来我将对这三种实现进行基准测试。


      性能比较


      在实现了字符串反转方法之后,我们测试下是三种方法的性能如何

      因此,我们开始进行一些基准测试:


      >>>importtimeit
      >>>s='abcdefghijklmnopqrstuvwxyz'*10
      >>>timeit.repeat(lambda:reverse_string1(s))
      [0.6848115339962533,0.7366074129968183,0.7358982900041156]
      >>>timeit.repeat(lambda:reverse_string2(s))
      [5.514941683999496,5.339547180992668,5.319950777004124]
      >>>timeit.repeat(lambda:reverse_string3(s))
      [48.74324739299482,48.637329410004895,49.223478018000606]

      汇总成表格形式:

      算法

      执行时间处理时间

      慢一点

      切片

      0.72

      1

      反向+加入

      5.39

      7.5

      经典

      48.87

      67.9

      由此可见,这三种实现之间存在巨大的性能差距

      切片是最快的方法,reversed()比切片慢8倍,而“经典”就地算法在该基准测试中要慢71倍!


      总结:

      如果您想知道在Python中反转字符串的最佳方法是什么,我的答案是:“取决于情况”。就我个人而言,我喜欢这种reversed()方法,因为它是“自我记录”且相当快。

      但是,有一种观点认为,出于性能考虑,应使用快八倍的切片方法……

      相关文章
      |
      14天前
      |
      测试技术 数据库 Python
      Python装饰器实战:打造高效性能计时工具
      在数据分析中,处理大规模数据时,分析代码性能至关重要。本文介绍如何使用Python装饰器实现性能计时工具,在不改变现有代码的基础上,方便快速地测试函数执行时间。该方法具有侵入性小、复用性强、灵活度高等优点,有助于快速发现性能瓶颈并优化代码。通过设置循环次数参数,可以更准确地评估函数的平均执行时间,提升开发效率。
      89 61
      Python装饰器实战:打造高效性能计时工具
      |
      3月前
      |
      机器学习/深度学习 Python
      堆叠集成策略的原理、实现方法及Python应用。堆叠通过多层模型组合,先用不同基础模型生成预测,再用元学习器整合这些预测,提升模型性能
      本文深入探讨了堆叠集成策略的原理、实现方法及Python应用。堆叠通过多层模型组合,先用不同基础模型生成预测,再用元学习器整合这些预测,提升模型性能。文章详细介绍了堆叠的实现步骤,包括数据准备、基础模型训练、新训练集构建及元学习器训练,并讨论了其优缺点。
      129 3
      |
      10天前
      |
      人工智能 自然语言处理 算法
      随机的暴力美学蒙特卡洛方法 | python小知识
      蒙特卡洛方法是一种基于随机采样的计算算法,广泛应用于物理学、金融、工程等领域。它通过重复随机采样来解决复杂问题,尤其适用于难以用解析方法求解的情况。该方法起源于二战期间的曼哈顿计划,由斯坦尼斯拉夫·乌拉姆等人提出。核心思想是通过大量随机样本来近似真实结果,如估算π值的经典示例。蒙特卡洛树搜索(MCTS)是其高级应用,常用于游戏AI和决策优化。Python中可通过简单代码实现蒙特卡洛方法,展示其在文本生成等领域的潜力。随着计算能力提升,蒙特卡洛方法的应用范围不断扩大,成为处理不确定性和复杂系统的重要工具。
      49 21
      |
      8天前
      |
      数据挖掘 数据处理 开发者
      Python3 自定义排序详解:方法与示例
      Python的排序功能强大且灵活,主要通过`sorted()`函数和列表的`sort()`方法实现。两者均支持`key`参数自定义排序规则。本文详细介绍了基础排序、按字符串长度或元组元素排序、降序排序、多条件排序及使用`lambda`表达式和`functools.cmp_to_key`进行复杂排序。通过示例展示了如何对简单数据类型、字典、类对象及复杂数据结构(如列车信息)进行排序。掌握这些技巧可以显著提升数据处理能力,为编程提供更强大的支持。
      24 10
      |
      3月前
      |
      Python
      在 Python 中,如何将日期时间类型转换为字符串?
      在 Python 中,如何将日期时间类型转换为字符串?
      141 64
      |
      14天前
      |
      存储 人工智能 Python
      [oeasy]python061_如何接收输入_input函数_字符串_str_容器_ 输入输出
      本文介绍了Python中如何使用`input()`函数接收用户输入。`input()`函数可以从标准输入流获取字符串,并将其赋值给变量。通过键盘输入的值可以实时赋予变量,实现动态输入。为了更好地理解其用法,文中通过实例演示了如何接收用户输入并存储在变量中,还介绍了`input()`函数的参数`prompt`,用于提供输入提示信息。最后总结了`input()`函数的核心功能及其应用场景。更多内容可参考蓝桥、GitHub和Gitee上的相关教程。
      11 0
      |
      2月前
      |
      安全
      Python-打印99乘法表的两种方法
      本文详细介绍了两种实现99乘法表的方法:使用`while`循环和`for`循环。每种方法都包括了步骤解析、代码演示及优缺点分析。文章旨在帮助编程初学者理解和掌握循环结构的应用,内容通俗易懂,适合编程新手阅读。博主表示欢迎读者反馈,共同进步。
      |
      2月前
      |
      存储 测试技术 Python
      Python 中别再用 ‘+‘ 拼接字符串了!
      通过选择合适的字符串拼接方法,可以显著提升 Python 代码的效率和可读性。在实际开发中,根据具体需求和场景选择最佳的方法,避免不必要的性能损失。
      58 5
      |
      2月前
      |
      Python
      使用Python计算字符串的SHA-256散列值
      使用Python计算字符串的SHA-256散列值
      59 7
      |
      2月前
      |
      JSON 安全 API
      Python调用API接口的方法
      Python调用API接口的方法
      321 5

      热门文章

      最新文章