侯体宗的博客
  • 首页
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

python 性能提升的几种方法

Python  /  管理员 发布于 7年前   199

关于python 性能提升的一些方案。

一、函数调用优化(空间跨度,避免访问内存)

 程序的优化核心点在于尽量减少操作跨度,包括代码执行时间上的跨度以及内存中空间跨度。

1.大数据求和,使用sum

a = range(100000)%timeit -n 10 sum(a)10 loops, best of 3: 3.15 ms per loop%%timeit  ...: s = 0  ...: for i in a:  ...:  s += i  ...:100 loops, best of 3: 6.93 ms per loop

2.小数据求和,避免使用sum

%timeit -n 1000 s = a + b + c + d + e + f + g + h + i + j + k # 数据量较小时直接累加更快1000 loops, best of 3: 571 ns per loop%timeit -n 1000 s = sum([a,b,c,d,e,f,g,h,i,j,k]) # 小数据量调用 sum 函数,空间效率降低1000 loops, best of 3: 669 ns per loop

结论:大数据求和sum效率高,小数据求和直接累加效率高。

二、for循环优化之取元素(使用栈或寄存器,避免访问内存)

for lst in [(1, 2, 3), (4, 5, 6)]: # lst 索引需要额外开销  pass

 应尽量避免使用索引。

for a, b, c in [(1, 2, 3), (4, 5, 6)]: # better  pass

相当于给每一个元素直接赋值。

def force(): lst = range(4) for a1 in [1, 2]:   for a2 in lst:     for a3 in lst:       for b1 in lst:         for b2 in lst:           for b3 in lst: for c1 in lst:   for c2 in lst:     for c3 in lst:       for d1 in lst:         yield (a1, a2, a3, b1, b2, b3, c1, c2, c3, d1)          %%timeit -n 10for t in force():  sum([t[0], t[1], t[2], t[3], t[4], t[5], t[6], t[7], t[8], t[9]])10 loops, best of 3: 465 ms per loop%%timeit -n 10for a1, a2, a3, b1, b2, b3, c1, c2, c3, d1 in force():  sum([a1, a2, a3, b1, b2, b3, c1, c2, c3, d1])10 loops, best of 3: 360 ms per loop

三、生成器优化(查表代替运算)

def force(start, end): # 用于密码暴力破解程序  for i in range(start, end):    now = i    sublst = []    for j in range(10):      sublst.append(i % 10) # 除法运算开销较大,比乘法大      i //= 10    sublst.reverse()    yield(tuple(sublst), now)
def force(): # better lst = range(5) for a1 in [1]:   for a2 in lst:     for a3 in lst:       for b1 in lst:         for b2 in lst:           for b3 in lst: for c1 in lst:   for c2 in lst:     for c3 in lst:       for d1 in lst:         yield (a1, a2, a3, b1, b2, b3, c1, c2, c3, d1)  
r0 = [1, 2] # 可读性与灵活性r1 = range(10)r2 = r3 = r4 = r5 = r6 = r7 = r8 = r9 = r1force = ((a0, a1, a2, a3, a4, a5, a6, a7, a8, a9)      for a0 in r0 for a1 in r1 for a2 in r2 for a3 in r3 for a4 in r4      for a5 in r5 for a6 in r6 for a7 in r7 for a8 in r8 for a9 in r9)

 四、幂运算优化(pow(x,y,z)) 

def isprime(n):  if n & 1 == 0:    return False  k, q = find_kq(n)  a = randint(1, n - 1)  if pow(a, q, n) == 1: # 比使用 a ** q % n 运算优化数倍    return True  for j in range(k):    if pow(a, pow(2, j) * q, n) == n - 1: # a **((2 ** j) * q) % n      return True  return False

 结论:pow(x,y,z)优于x**y%z.

 五、除法运算优化

In [1]: from random import getrandbits In [2]: x = getrandbits(4096) In [3]: y = getrandbits(2048) In [4]: %timeit -n 10000 q, r = divmod(x, y)10000 loops, best of 3: 10.7 us per loop In [5]: %timeit -n 10000 q, r = x//y, x % y10000 loops, best of 3: 21.2 us per loop

 结论:divmod优于//和%。

 六、优化算法时间复杂度  

算法的时间复杂度对程序的执行效率影响最大,在python中可以选择合适的数据结构来优化时间复杂度,如list和set查找某一个元素的时间复杂度分别是O(n)和O(1)。不同场景有不同的优化方式,总的来说,一般有分治,分支定界、贪心动态规划等思想。

七、合理使用copy和deepcopy  

对于dict和list等数据结构的对象,直接赋值使用的是引用的方式。而有些情况下需要复制整个对象,这时可以使用copy包里的copy和deepcopy,这两个函数的不同之处在于deepcopy是递归复制的。效率不同:

In [23]: import copyIn [24]: %timeit -n 10 copy.copy(a)10 loops, best of 3: 606 ns per loopIn [25]: %timeit -n 10 copy.deepcopy(a)10 loops, best of 3: 1.17 us per loop

 timeit后面的-n表示运行的次数,后两行对应的是两个timeit的输出,下同。由此可见后者慢一个数量级。

 关于copy的一个例子:

>>> lists = [[]] * 3>>> lists[[], [], []]>>> lists[0].append(3)>>> lists[[3], [3], [3]]

 发生的事情是这样的,[[]]是包含一个空列表的只有一个元素的列表,所以[[]] * 3的所有三个元素都是(指向)这个空列表。修改lists的任何元素都修改这个列表。修改效率高。

 八、使用dict或set查找元素

python 字典和集合都是使用hash表来实现(类似c++标准库unordered_map),查找元素的时间复杂度是O(1)。

In [1]: r = range(10**7)In [2]: s = set(r) # 占用 588MB 内存In [3]: d = dict((i, 1) for i in r) # 占用 716MB 内存In [4]: %timeit -n 10000 (10**7) - 1 in r10000 loops, best of 3: 291 ns per loopIn [5]: %timeit -n 10000 (10**7) - 1 in s10000 loops, best of 3: 121 ns per loopIn [6]: %timeit -n 10000 (10**7) - 1 in d10000 loops, best of 3: 111 ns per loop

结论:set 的内存占用量最小,dict运行时间最短。

九、合理使用(generator)和yield(节省内存)

In [1]: %timeit -n 10 a = (i for i in range(10**7)) # 生成器通常遍历更高效10 loops, best of 3: 933 ns per loopIn [2]: %timeit -n 10 a = [i for i in range(10**7)]10 loops, best of 3: 916 ms per loopIn [1]: %timeit -n 10 for x in (i for i in range(10**7)): pass10 loops, best of 3: 749 ms per loopIn [2]: %timeit -n 10 for x in [i for i in range(10**7)]: pass10 loops, best of 3: 1.05 s per loop

结论:尽量使用生成器去遍历。

以上就是对python 性能提升的一些方案,后续继续补充,需要的可以看下。

您可能感兴趣的文章:

  • Python判断列表是否已排序的各种方法及其性能分析
  • 实例探究Python以并发方式编写高性能端口扫描器的方法
  • 举例讲解Python面相对象编程中对象的属性与类的方法
  • 菜鸟使用python实现正则检测密码合法性
  • Python Property属性的2种用法
  • Python编程之属性和方法实例详解
  • python通过函数属性实现全局变量的方法


  • 上一条:
    python 实现网上商城,转账,存取款等功能的信用卡系统
    下一条:
    浅谈Python 对象内存占用
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 在python语言中Flask框架的学习及简单功能示例(0个评论)
    • 在Python语言中实现GUI全屏倒计时代码示例(0个评论)
    • Python + zipfile库实现zip文件解压自动化脚本示例(0个评论)
    • python爬虫BeautifulSoup快速抓取网站图片(1个评论)
    • vscode 配置 python3开发环境的方法(0个评论)
    • 近期文章
    • 在go语言中实现字符串可逆性压缩及解压缩功能(0个评论)
    • 使用go + gin + jwt + qrcode实现网站生成登录二维码在app中扫码登录功能(0个评论)
    • 在windows10中升级go版本至1.24后LiteIDE的Ctrl+左击无法跳转问题解决方案(0个评论)
    • 智能合约Solidity学习CryptoZombie第四课:僵尸作战系统(0个评论)
    • 智能合约Solidity学习CryptoZombie第三课:组建僵尸军队(高级Solidity理论)(0个评论)
    • 智能合约Solidity学习CryptoZombie第二课:让你的僵尸猎食(0个评论)
    • 智能合约Solidity学习CryptoZombie第一课:生成一只你的僵尸(0个评论)
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2018-04
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2022-01
    • 2023-07
    • 2023-10
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客