草庐IT

python - 优化Python代码

coder 2023-05-26 原文

我一直在研究interviewstreet.com上的一个编码难题,我遇到了一个效率问题。有人能建议我在哪里修改代码以使其更快、更高效吗?
Here's the code
Here's the problem statement if you're interested

最佳答案

如果您的问题是一般优化python代码(我认为应该是;),那么您可以做各种无遗嘱的事情,但首先:
您可能不应该过分优化Python代码!如果您正在使用最快的算法来解决您试图解决的问题,而Python的速度不够快,那么您可能应该使用不同的语言。
也就是说,您可以采用几种方法(因为有时您确实希望加快Python代码的速度):
配置文件(请先执行此操作!)
分析python代码有很多种方法,但我要提到两种方法:cProfile (or profile) modulePyCallGraph
碳纤维
这是你实际应该使用的,尽管解释结果可能有点令人生畏。
它通过记录每个函数的输入或退出时间以及调用函数是什么(以及跟踪异常)来工作。
您可以这样在cprofile中运行函数:

import cProfile
cProfile.run('myFunction()', 'myFunction.profile')

然后查看结果:
import pstats
stats = pstats.Stats('myFunction.profile')
stats.strip_dirs().sort_stats('time').print_stats()

这将向您展示大部分时间用于哪些功能。
比重计
PyCallGraph提供了一种分析python程序的最漂亮和最简单的方法——这是了解程序时间在哪里花费的一个很好的介绍,但是它增加了大量的执行开销。
要运行pycallgraph:
pycallgraph graphviz ./myprogram.py

简单!您将得到一个PNG图形图像作为输出(可能在一段时间之后…)
使用库
如果您试图在python中做一些模块已经存在的事情(甚至可能在标准库中),那么使用该模块!
大多数标准库模块都是用C语言编写的,它们的执行速度将比类似的python实现(比如,bisection search)快数百倍。
让翻译尽可能多地做你的工作
解释器会为您做一些事情,比如循环。真正地?对!您可以使用mapreducefilter关键字显著加速紧密循环:
考虑:
for x in xrange(0, 100):
    doSomethingWithX(x)

VS:
map(doSomethingWithX, xrange(0,100))

很明显,这可能更快,因为解释器只需要处理一个语句,而不是两个,但这有点含糊…事实上,这种速度更快有两个原因:
所有流控制(我们已经完成了循环…)都在解释器中完成。
DoSomethingWithX函数名只解析一次
在for循环中,每次循环python都必须精确地检查doSomethingWithX函数的位置!即使有了缓存,这也有点开销。
记住,python是一种解释语言
(请注意,本节实际上是关于微小的优化,您不应该让这些优化影响您正常的、可读的编码风格!)
如果您来自编译语言(如C或Fortran)编程的背景,那么关于不同python语句的性能的一些事情可能会令人惊讶:
便宜,贵
如果您有这样的代码:
if somethingcrazy_happened:
     uhOhBetterDoSomething()
else:
     doWhatWeNormallyDo()

如果发生了疯狂的事情,try:会抛出一个异常,那么这样安排代码会更快:
try:
    doWhatWeNormallyDo()
except SomethingCrazy:
    uhOhBetterDoSomething()

为什么?好吧,解释器可以直接进入并开始做你通常做的事情;在第一种情况下,解释器必须在每次执行if语句时做一个符号查找,因为名称可能引用自上次执行语句以来不同的东西!(尤其是当ifdoWhatWeNormallyDo()时,名称查找是非常重要的)。
你是说谁??
由于名称查找的成本,最好在函数中缓存全局值,并将简单的布尔测试烘焙到如下函数中:
未优化的功能:
def foo():
    if condition_that_rarely_changes:
         doSomething()
    else:
         doSomethingElse()

优化的方法,而不是使用变量,利用这样一个事实:无论如何,解释器都在对函数进行名称查找!
当条件成立时:
foo = doSomething # now foo() calls doSomething()

当条件变为错误时:
foo = doSomethingElse # now foo() calls doSomethingElse()

派皮
PyPy是用python编写的python实现。当然,这意味着它将以无限慢的速度运行代码?不,Pypy实际上使用实时编译器(just-in-time compiler,JIT)来运行python程序。
如果您不使用任何外部库(或者您确实使用的库是compatible with PyPy),那么这是一种非常简单的方法(几乎可以肯定)加速程序中的重复任务。
基本上,JIT可以生成代码来完成Python解释器所做的工作,但速度要快得多,因为它是为单个案例生成的,而不是必须处理所有可能的合法Python表达式。
下一步看哪里
当然,您首先应该考虑的是改进算法和数据结构,并考虑诸如缓存之类的问题,甚至是您是否需要在一开始就做这么多事情,但无论如何:
python.org wiki的This page提供了很多关于如何加速python代码的信息,尽管其中一些已经过时了。
这里是关于优化循环的BDFL himself主题。
有很多事情,即使是从我自己有限的经验中,我错过了,但这个答案已经足够长了!
这都是基于我最近对一些不够快的python代码的体验,我想再次强调,我并不认为我所建议的是一个好主意,有时候,你必须……

关于python - 优化Python代码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7165465/

有关python - 优化Python代码的更多相关文章

  1. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  2. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  3. ruby-on-rails - Rails 源代码 : initialize hash in a weird way? - 2

    在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has

  4. ruby-on-rails - 浏览 Ruby 源代码 - 2

    我的主要目标是能够完全理解我正在使用的库/gem。我尝试在Github上从头到尾阅读源代码,但这真的很难。我认为更有趣、更温和的踏脚石就是在使用时阅读每个库/gem方法的源代码。例如,我想知道RubyonRails中的redirect_to方法是如何工作的:如何查找redirect_to方法的源代码?我知道在pry中我可以执行类似show-methodmethod的操作,但我如何才能对Rails框架中的方法执行此操作?您对我如何更好地理解Gem及其API有什么建议吗?仅仅阅读源代码似乎真的很难,尤其是对于框架。谢谢! 最佳答案 Ru

  5. ruby - 模块嵌套代码风格偏好 - 2

    我的假设是moduleAmoduleBendend和moduleA::Bend是一样的。我能够从thisblog找到解决方案,thisSOthread和andthisSOthread.为什么以及什么时候应该更喜欢紧凑语法A::B而不是另一个,因为它显然有一个缺点?我有一种直觉,它可能与性能有关,因为在更多命名空间中查找常量需要更多计算。但是我无法通过对普通类进行基准测试来验证这一点。 最佳答案 这两种写作方法经常被混淆。首先要说的是,据我所知,没有可衡量的性能差异。(在下面的书面示例中不断查找)最明显的区别,可能也是最著名的,是你的

  6. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

  7. ruby - Net::HTTP 获取源代码和状态 - 2

    我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur

  8. Python 相当于 Perl/Ruby ||= - 2

    这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Pythonconditionalassignmentoperator对于这样一个简单的问题表示歉意,但是谷歌搜索||=并不是很有帮助;)Python中是否有与Ruby和Perl中的||=语句等效的语句?例如:foo="hey"foo||="what"#assignfooifit'sundefined#fooisstill"hey"bar||="yeah"#baris"yeah"另外,类似这样的东西的通用术语是什么?条件分配是我的第一个猜测,但Wikipediapage跟我想的不太一样。

  9. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  10. 程序员如何提高代码能力? - 2

    前言作为一名程序员,自己的本质工作就是做程序开发,那么程序开发的时候最直接的体现就是代码,检验一个程序员技术水平的一个核心环节就是开发时候的代码能力。众所周知,程序开发的水平提升是一个循序渐进的过程,每一位程序员都是从“菜鸟”变成“大神”的,所以程序员在程序开发过程中的代码能力也是根据平时开发中的业务实践来积累和提升的。提高代码能力核心要素程序员要想提高自身代码能力,尤其是新晋程序员的代码能力有很大的提升空间的时候,需要针对性的去提高自己的代码能力。提高代码能力其实有几个比较关键的点,只要把握住这些方面,就能很好的、快速的提高自己的一部分代码能力。1、多去阅读开源项目,如有机会可以亲自参与开源

随机推荐