`
daiyuok
  • 浏览: 26541 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Python性能优化

 
阅读更多

1. 使用内建函数:

    你可以用Python写出高效的代码,但很难击败内建函数. 经查证. 他们非常快速.


2.使用join()连接字符串.

      你可以使用 "+" 来连接字符串. 但由于string在Python中是不可变的,每一个"+"操作都会创建一个新的字符串并复制旧内容. 常见用法是使用Python的数组模块单个的修改字符;当完成的时候,使用 join() 函数创建最终字符串. 
      >>> #This is good to glue a large number of strings 
      >>> for chunk in input(): 
      >>>    my_string.join(chunk) 

3. 使用Python多重赋值,交换变量 
    这在Python中即优雅又快速: 
      >>> x, y = y, x 
      这样很慢: 
      >>> temp = x 
      >>> x = y 
      >>> y = temp           

4. 尽量使用局部变量 
    Python 检索局部变量比检索全局变量快. 这意味着,避免 "global" 关键字. 

5. 尽量使用 "in" 
      使用 "in" 关键字. 简洁而快速. 
      >>> for key in sequence: 
      >>>     print “found” 

6. 使用延迟加载加速 
      將 "import" 声明移入函数中,仅在需要的时候导入. 换句话说,如果某些模块不需马上使用,稍后导入他们. 例如,你不必在一开使就导入大量模块而加速程序启动. 该技术不能提高整体性能. 但它可以帮助你更均衡的分配模块的加载时间. 

7. 为无限循环使用 "while 1" 
    有时候在程序中你需一个无限循环.(例如一个监听套接字的实例) 尽管 "while true" 能完成同样的事, 但 "while 1" 是单步运算. 这招能提高你的Python性能. 
       >>> while 1: 
      >>>    #do stuff, faster with while 1 
      >>> while True: 

      >>>    # do stuff, slower with wile True


8. 使用list comprehension 
    从Python 2.0 开始,你可以使用 list comprehension 取代大量的 "for" 和 "while" 块. 使用List comprehension通常更快,Python解析器能在循环中发现它是一个可预测的模式而被优化.额外好处是,list comprehension更具可读性(函数式编程),并在大多数情况下,它可以节省一个额外的计数变量。例如,让我们计算1到10之间的偶数个数: 
      >>> # the good way to iterate a range 
      >>> evens = [ i for i in range(10) if i%2 == 0] 
      >>> [0, 2, 4, 6, 8] 
      >>> # the following is not so Pythonic 
      >>> i = 0 
      >>> evens = [] 
      >>> while i < 10: 
      >>>    if i %2 == 0: evens.append(i) 
      >>>    i += 1 
      >>> [0, 2, 4, 6, 8] 

9. 使用xrange()处理长序列: 
    这样可为你节省大量的系统内存,因为xrange()在序列中每次调用只产生一个整数元素。而相反 range(),它將直接给你一个完整的元素列表,用于循环时会有不必要的开销。 

10. 使用 Python generator: 
      这也可以节省内存和提高性能。例如一个视频流,你可以一个一个字节块的发送,而不是整个流。例如,      
      >>> chunk = ( 1000 * i for i in xrange(1000)) 
      >>> chunk 
      <generator object <genexpr> at 0x7f65d90dcaa0> 
      >>> chunk.next() 
      0 
      >>> chunk.next() 
      1000 
      >>> chunk.next() 
      2000 

11. 了解itertools模块: 
      该模块对迭代和组合是非常有效的。让我们生成一个列表[1,2,3]的所有排列组合,仅需三行Python代码: 
      >>> import itertools 
      >>> iter = itertools.permutations([1,2,3]) 
      >>> list(iter) 
      [(1, 2, 3), (1, 3, 2), (2, 1, 3), (2, 3, 1), (3, 1, 2), (3, 2, 1)] 

12. 学习bisect模块保持列表排序: 
      这是一个免费的二分查找实现和快速插入有序序列的工具。也就是说,你可以使用: 
      >>> import bisect 
      >>> bisect.insort(list, element) 
      你已將一个元素插入列表中, 而你不需要再次调用 sort() 来保持容器的排序, 因为这在长序列中这会非常昂贵. 

13. 理解Python列表,实际上是一个数组: 
      Python中的列表实现并不是以人们通常谈论的计算机科学中的普通单链表实现的。Python中的列表是一个数组。也就是说,你可以以常量时间O(1) 检索列表的某个元素,而不需要从头开始搜索。这有什么意义呢? Python开发人员使用列表对象insert()时, 需三思. 例如:>>> list.insert(0,item) 
      在列表的前面插入一个元素效率不高, 因为列表中的所有后续下标不得不改变. 然而,您可以使用list.append()在列表的尾端有效添加元素. 挑先deque,如果你想快速的在两插入或时。它是快速的,因为在Python中的deque用双链表实现。不再多说。 :) 

14. 使用dict 和 set 测试成员:

      检查一个元素是在dicitonary或set是否存在 这在Python中非常快的。这是因为dict和set使用哈希表来实现。查找效率可以达到O(1)。因此,如果您需要经常检查成员,使用 set 或 dict做为你的容器. 
      >>> mylist = ['a', 'b', 'c'] #Slower, check membership with list: 
      >>> ‘c’ in mylist 
      >>> True 
      >>> myset = set(['a', 'b', 'c']) # Faster, check membership with set: 
      >>> ‘c’ in myset: 
      >>> True 

15. 使用Schwartzian Transform 的 sort():


      原生的list.sort()函数是非常快的。 Python会按自然顺序排序列表。有时,你需要非自然顺序的排序。例如,你要根据服务器位置排序的IP地址。 Python支持自定义的比较,你可以使用list.sort(CMP()),这会比list.sort()慢,因为增加了函数调用的开销。如果性能有问 题,你可以申请Guttman-Rosler Transform,基于Schwartzian Transform. 它只对实际的要用的算法有兴趣,它的简要工作原理是,你可以变换列表,并调用Python内置list.sort() - > 更快,而无需使用list.sort(CMP() )->慢。 


16. Python装饰器缓存结果: 
      “@”符号是Python的装饰语法。它不只用于追查,锁或日志。你可以装饰一个Python函数,记住调用结果供后续使用。这种技术被称为memoization的。下面是一个例子: 

      >>> from functools import wraps 
      >>> def memo(f): 
      >>>    cache = { } 
      >>>    @wraps(f) 
      >>>    def  wrap(*arg): 
      >>>        if arg not in cache: cache['arg'] = f(*arg) 
      >>>        return cache['arg'] 
      >>>    return wrap 
      我们也可以对 Fibonacci 函数使用装饰器: 
      >>> @memo 
      >>> def fib(i): 
      >>>    if i < 2: return 1 
      >>>    return fib(i-1) + fib(i-2) 
    
     这里的关键思想是:增强函数(装饰)函数,记住每个已经计算的Fibonacci值;如果它们在缓存中,就不需要再计算了. 

17. 理解Python的GIL(全局解释器锁): 
      GIL是必要的,因为CPython的内存管理是非线程安全的。你不能简单地创建多个线程,并希望Python能在多核心的机器上运行得更快。这是因为 GIL將会防止多个原生线程同时执行Python字节码。换句话说,GIL將序列化您的所有线程。然而,您可以使用线程管理多个派生进程加速程序,这些程 序独立的运行于你的Python代码外。 

18. 像熟悉文档一样的熟悉Python源代码: 
      Python有些模块为了性能使用C实现。当性能至关重要而官方文档不足时,可以自由探索源代码。你可以找到底层的数据结构和算法。 Python的源码库就是一个很棒的地方:http://svn.python.org/view/python/trunk/Modules 

结论: 
这些不能替代大脑思考. 打开引擎盖充分了解是开发者的职责,使得他们不会快速拼凑出一个垃圾设计. 本文的Python建议可以帮助你获得好的性能. 如果速度还不够快, Python將需要借助外力:分析和运行外部代码.我们將在本文的第二部分中涉及. 

第二部分 

有益的提醒,静态编译的代码仍然重要. 仅例举几例, Chrome,Firefox,MySQL,MS Office 和 Photoshop都是高度优化的软件,我们每天都在使用. Python作为解析语言,很明显不适合. 不能单靠Python来满足那些性能是首要指示的领域.  这就是为什么Python支持让你接触底层裸机基础设施的原因, 将更繁重的工作代理给更快的语言如C. 这高性能计算和嵌入式编程中是关键的功能. Python性能鸡汤第一部分讨论了怎样高效的使用Python. 在第二部分, 我们將涉及监控和扩展Python.

1. 首先, 拒绝调优诱惑


    调优给你的代码增加复杂性. 集成其它语言之前, 请检查下面的列表. 如果你的算法是"足够好", 优化就没那么迫切了. 
    1. 你做了性能测试报告吗? 
    2. 你能减少硬盘的 I/O 访问吗? 
    3. 你能减少网络 I/O 访问吗? 
    4. 你能升级硬件吗? 
    5. 你是为其它开发者编译库吗? 
    6.你的第三方库软件是最新版吗? 

2. 使用工具监控代码, 而不是直觉 
        速度的问题可能很微妙, 所以不要依赖于直觉. 感谢 "cprofiles" 模块, 通过简单的运行你就可以监控Python代码 
“python -m cProfile myprogram.py” 
 

        我们写了个测试程序. 基于黑盒监控. 这里的瓶颈是 "very_slow()" 函数调用. 我们还可以看到 "fast()" 和 "slow()"都被调用200次. 这意味着, 如果我们可以改善 "fast()" 和 "slow()" 函数, 我们可以获得全面的性能提升. cprofiles 模块也可以在运行时导入. 这对于检查长时间运行的进程非常有用.


3. 审查时间复杂度 
        控制以后, 提供一个基本的算法性能分析. 恒定时间是理想值. 对数时间复度是稳定的. 阶乘复杂度很难扩展. 

      O(1) -> O(lg n) -> O(n lg n) -> O(n^2) -> O(n^3) -> O(n^k) -> O(k^n) -> O(n!) 

4. 使用第三方包 

        有很多为Python设计的高性能的第三方库和工具. 下面是一些有用的加速包的简短列表. 

    1. NumPy: 一个开源的相当于MatLab的包 
    2. SciPy: 另一个数值处理库 
    3. GPULib: 使用GPUs加速代码 
    4. PyPy: 使用 just-in-time 编译器优化Python代码 
    5. Cython: 將Python优码转成C 

    6. ShedSkin: 將Python代码转成C++


5. 使用multiprocessing模块实现真正的并发 
      因为GIL会序列化线程, Python中的多线程不能在多核机器和集群中加速. 因此Python提供了multiprocessing模块, 可以派生额外的进程代替线程, 跳出GIL的限制. 此外, 你也可以在外部C代码中结合该建议, 使得程序更快. 

        注意, 进程的开销通常比线程昂贵, 因为线程自动共享内存地址空间和文件描述符. 意味着, 创建进程比创建线程会花费更多, 也可能花费更多内存. 这点在你计算使用多处理器时要牢记.


6. 本地代码 

        好了, 现在你决定为了性能使用本地代码. 在标准的ctypes模块中, 你可以直接加载已编程的二进制库(.dll 或 .so文件)到Python中, 无需担心编写C/C++代码或构建依赖. 例如, 我们可以写个程序加载libc来生成随机数. 

       然而, 绑定ctypes的开销是非轻量级的. 你可以认为ctypes是一个粘合操作系库函数或者硬件设备驱动的胶水. 有几个如 SWIG, Cython和Boost 此类Python直接植入的库的调用比ctypes开销要低. Python支持面向对象特性, 如类和继承. 正如我们看到的例子, 我们可以保留常规的C++代码, 稍后导入. 这里的主要工作是编写一个包装器 (行 10~18).


总结: 

     我希望这些Python建议能让你成为一个更好的开发者. 最后, 我需要指出, 追求性能极限是一个有趣的游戏, 而过度优化就会变成嘲弄了. 虽然Python授予你与C接口无缝集成的能力, 你必须问自己你花数小时的艰辛优化工作用户是否买帐. 另一方面, 牺牲代码的可维护性换取几毫秒的提升是否值得. 团队中的成员常常会感谢你编写了简洁的代码. 尽量贴近Python的方式, 因为人生苦短

分享到:
评论

相关推荐

    python性能优化实践1

    【Python性能优化实践1】 在Python编程中,性能优化是一个重要的方面,特别是在处理大量数据或运行计算密集型任务时。本文将探讨两个关键的优化技巧:避免重复计算和理解真假多线程的概念。 首先,避免重复计算是...

    python性能优化的技巧1

    【Python性能优化技巧1】 在Python编程中,性能优化是一个重要的方面,特别是在处理大量数据或对性能有较高要求的应用中。以下是一些Python性能优化的关键技巧和原理。 1. **少造轮子** - Python的标准库是经过...

    Python性能优化:掌握性能分析工具的实战指南

    # Python性能优化:掌握性能分析工具的实战指南 Python 是一种高级编程语言,因其简洁的语法和可读性被广泛采用。Python 的设计哲学强调代码的简洁与清晰,使其成为初学者的理想选择,并且同样受到专业开发者的青睐...

    Python性能优化-TaoBeier@meitu.pdf

    在探讨Python性能优化的过程中,首先要了解性能优化的前提条件。首先,优化工作必须建立在程序本身是正确运行的基础之上,没有意义的性能优化可能会导致错误的代码运行更快,从而加大bug修复的难度。其次,性能优化...

    Python 代码性能优化技巧

    选择了脚本语言就要忍受其速度,这...如何进行 Python 性能优化,是本文探讨的主要问题。本文会涉及常见的代码优化方法,性能优化工具的使用以及如何诊断代码的性能瓶颈等内容,希望可以给 Python 开发人员一定的参考。

    Python性能优化经验谈.zip

    本资料"Python性能优化经验谈"将深入探讨如何提升Python程序的运行效率,使其在资源消耗上更加高效。以下是对PDF文件内容的详细概述: 1. **Python解释器的选择与优化** - CPython是Python的默认解释器,但其解释...

    python性能优化实例练习.rar

    python性能优化实例练习.rar

    Python性能分析与优化1

    书中详尽的解释和实例将帮助你掌握Python性能优化的技巧,提高代码执行效率,尤其是在处理大数据、文本分析和服务器响应等高负载场景下。 译者序部分以一种生动的方式阐述了并行计算的概念,通过对比太阳普照地球的...

    Python性能优化技巧

    Python使许多编程任务变得简单,但是对于很关键的任务并不总是提供最好的性能。使用C、C++或者机器语言扩展包来执行关键任务能极大改善性能。这些包是依赖于平台的,也就是说,你必须使用特定的、与你使用的平台相关...

    Python 代码性能优化技巧分享

    如何进行 Python 性能优化,是本文探讨的主要问题。本文会涉及常见的代码优化方法,性能优化工具的使用以及如何诊断代码的性能瓶颈等内容,希望可以给 Python 开发人员一定的参考。 Python 代码优化常见技巧 代码...

    Python 性能优化技巧总结

    ### Python性能优化技巧详解 #### 一、引言 Python作为一种高级编程语言,在科学计算、数据分析、Web开发等领域有着广泛的应用。然而,由于其解释执行的特性,相比编译型语言如C或Java,Python在运行速度方面并不...

    Python高性能编程_python进阶_python高性能_源码.zip

    Python是一种广泛使用的高级编程语言...在"Python高性能编程"的源码中,可能包含了上述各个方面的示例和实践,通过学习和研究这些源码,可以加深对Python性能优化的理解,并将所学应用到实际项目中,提升代码执行效率。

    python 性能优化方法小结

    ### Python性能优化方法详解 Python作为一种广泛使用的高级编程语言,其简洁性和易读性深受开发者喜爱。然而,在处理大规模数据集或复杂计算任务时,Python的执行效率有时会成为瓶颈。因此,掌握Python性能优化技巧...

    python高性能

    《高性能Python》不仅是一本技术手册,更是一本深入探讨Python性能优化策略的指南书。通过对这些理论和实践技巧的学习,读者不仅可以提升自己的编程技能,还能更好地应对复杂多变的实际项目挑战。无论你是刚接触...

    Python教程-解释器及性能优化.pdf

    谈到性能优化,Psyco曾是一个流行的Python扩展模块,它可以即时优化代码,提升执行效率。然而,Psyco项目已停止维护,现在推荐使用PyPy作为替代。PyPy不仅提供了JIT编译,还在多种操作系统上表现优秀,它能够动态地...

    基于php和python的zabbix优化模板

    2. Python性能优化:针对Python脚本进行算法优化,减少资源消耗,提升执行速度。 3. 数据库优化:确保Zabbix数据库(通常是MySQL或PostgreSQL)的索引、缓存和查询优化。 综上所述,结合PHP和Python的Zabbix优化...

Global site tag (gtag.js) - Google Analytics