加速Python for循环的12种方法

python内建的一个常用功能是timeit模块。下面几节中我们将使用它来度量循环的当前性能和改进后的性能。
对于每种方法,我们通过运行测试来建立基线,该测试包括在10次测试运行中运行被测函数100k次(循环),然后计算每个循环的平均时间(以纳秒为单位,ns)。
几个简单方法
1、列表推导式
# baseline version (inefficient way) # calculating the power of numbers # without using list comprehension def test_01_v0(numbers):   output = []   for n in numbers:       output.append(n ** 2.5)   return output  # improved version # (using list comprehension) def test_01_v1(numbers):   output = [n ** 2.5 for n in numbers]   return output  
结果如下:
# summary of test results      baseline: 32.158 ns per loop      improved: 16.040 ns per loop % improvement: 50.1 %      speedup: 2.00x  
可以看到使用列表推导式可以得到2倍速的提高
2、在外部计算长度
如果需要依靠列表的长度进行迭代,请在for循环之外进行计算。
# baseline version (inefficient way) # (length calculation inside for loop) def test_02_v0(numbers):   output_list = []   for i in range(len(numbers)):     output_list.append(i * 2)   return output_list  # improved version # (length calculation outside for loop) def test_02_v1(numbers):   my_list_length = len(numbers)   output_list = []   for i in range(my_list_length):     output_list.append(i * 2)   return output_list  
通过将列表长度计算移出for循环,加速1.6倍,这个方法可能很少有人知道吧。
# summary of test results      baseline: 112.135 ns per loop      improved: 68.304 ns per loop % improvement: 39.1 %      speedup: 1.64x  
3、使用set
在使用for循环进行比较的情况下使用set。
# use for loops for nested lookups def test_03_v0(list_1, list_2):   # baseline version (inefficient way)   # (nested lookups using for loop)   common_items = []   for item in list_1:       if item in list_2:           common_items.append(item)   return common_items  def test_03_v1(list_1, list_2):   # improved version   # (sets to replace nested lookups)   s_1 = set(list_1)   s_2 = set(list_2)   output_list = []   common_items = s_1.intersection(s_2)   return common_items  
在使用嵌套for循环进行比较的情况下,使用set加速498x
# summary of test results      baseline: 9047.078 ns per loop      improved:   18.161 ns per loop % improvement: 99.8 %      speedup: 498.17x  
4、跳过不相关的迭代
避免冗余计算,即跳过不相关的迭代。
# example of inefficient code used to find # the first even square in a list of numbers def function_do_something(numbers):   for n in numbers:     square = n * n     if square % 2 == 0:         return square    return none  # no even square found  # example of improved code that # finds result without redundant computations def function_do_something_v1(numbers):   even_numbers = [i for n in numbers if n%2==0]   for n in even_numbers:     square = n * n     return square    return none  # no even square found  
这个方法要在设计for循环内容的时候进行代码设计,具体能提升多少可能根据实际情况不同:
# summary of test results      baseline: 16.912 ns per loop      improved: 8.697 ns per loop % improvement: 48.6 %      speedup: 1.94x  
5、代码合并
在某些情况下,直接将简单函数的代码合并到循环中可以提高代码的紧凑性和执行速度。
# example of inefficient code # loop that calls the is_prime function n times. def is_prime(n):   if n <= 1:     return false   for i in range(2, int(n**0.5) + 1):     if n % i == 0:       return false    return true  def test_05_v0(n):   # baseline version (inefficient way)   # (calls the is_prime function n times)   count = 0   for i in range(2, n + 1):     if is_prime(i):       count += 1   return count  def test_05_v1(n):   # improved version   # (inlines the logic of the is_prime function)   count = 0   for i in range(2, n + 1):     if i <= 1:       continue     for j in range(2, int(i**0.5) + 1):       if i % j == 0:         break     else:       count += 1   return count  
这样也可以提高1.3倍
# summary of test results      baseline: 1271.188 ns per loop      improved: 939.603 ns per loop % improvement: 26.1 %      speedup: 1.35x  
这是为什么呢?
调用函数涉及开销,例如在堆栈上推入和弹出变量、函数查找和参数传递。当一个简单的函数在循环中被重复调用时,函数调用的开销会增加并影响性能。所以将函数的代码直接内联到循环中可以消除这种开销,从而可能显著提高速度。
但是这里需要注意,平衡代码可读性和函数调用的频率是一个要考虑的问题。
一些小技巧
6 .避免重复
考虑避免重复计算,其中一些计算可能是多余的,并且会减慢代码的速度。相反,在适用的情况下考虑预计算。
def test_07_v0(n):   # example of inefficient code   # repetitive calculation within nested loop   result = 0   for i in range(n):     for j in range(n):       result += i * j   return result  def test_07_v1(n):   # example of improved code   # utilize precomputed values to help speedup   pv = [[i * j for j in range(n)] for i in range(n)]   result = 0   for i in range(n):     result += sum(pv[i][:i+1])   return result  
结果如下
# summary of test results      baseline: 139.146 ns per loop      improved: 92.325 ns per loop % improvement: 33.6 %      speedup: 1.51x  
7、使用generators
生成器支持延迟求值,也就是说,只有当你向它请求下一个值时,里面的表达式才会被求值,动态处理数据有助于减少内存使用并提高性能。尤其是大型数据集中
def test_08_v0(n):   # baseline version (inefficient way)   # (inefficiently calculates the nth fibonacci   # number using a list)   if n <= 1:     return n   f_list = [0, 1]   for i in range(2, n + 1):     f_list.append(f_list[i - 1] + f_list[i - 2])   return f_list[n]  def test_08_v1(n):   # improved version   # (efficiently calculates the nth fibonacci   # number using a generator)   a, b = 0, 1   for _ in range(n):     yield a     a, b = b, a + b  
可以看到提升很明显:
# summary of test results      baseline: 0.083 ns per loop      improved: 0.004 ns per loop % improvement: 95.5 %      speedup: 22.06x  
8、map()函数
使用python内置的map()函数。它允许在不使用显式for循环的情况下处理和转换可迭代对象中的所有项。
def some_function_x(x):   # this would normally be a function containing application logic   # which required it to be made into a separate function   # (for the purpose of this test, just calculate and return the square)   return x**2  def test_09_v0(numbers):   # baseline version (inefficient way)   output = []   for i in numbers:     output.append(some_function_x(i))    return output  def test_09_v1(numbers):   # improved version   # (using python's built-in map() function)   output = map(some_function_x, numbers)   return output  
使用python内置的map()函数代替显式的for循环加速了970x。
# summary of test results      baseline: 4.402 ns per loop      improved: 0.005 ns per loop % improvement: 99.9 %      speedup: 970.69x  
这是为什么呢?
map()函数是用c语言编写的,并且经过了高度优化,因此它的内部隐含循环比常规的python for循环要高效得多。因此速度加快了,或者可以说python还是太慢,哈。
9、使用memoization
记忆优化算法的思想是缓存(或“记忆”)昂贵的函数调用的结果,并在出现相同的输入时返回它们。它可以减少冗余计算,加快程序速度。
首先是低效的版本。
# example of inefficient code def fibonacci(n):   if n == 0:     return 0   elif n == 1:     return 1   return fibonacci(n - 1) + fibonacci(n-2)  def test_10_v0(list_of_numbers):   output = []   for i in numbers:     output.append(fibonacci(i))    return output  
然后我们使用python的内置functools的lru_cache函数。
# example of efficient code # using python's functools' lru_cache function import functools  @functools.lru_cache() def fibonacci_v2(n):   if n == 0:     return 0   elif n == 1:     return 1   return fibonacci_v2(n - 1) + fibonacci_v2(n-2)  def _test_10_v1(numbers):   output = []   for i in numbers:     output.append(fibonacci_v2(i))    return output  
结果如下:
# summary of test results      baseline: 63.664 ns per loop      improved: 1.104 ns per loop % improvement: 98.3 %      speedup: 57.69x  
使用python的内置functools的lru_cache函数使用memoization加速57x。
lru_cache函数是如何实现的?
“lru”是“least recently used”的缩写。lru_cache是一个装饰器,可以应用于函数以启用memoization。它将最近函数调用的结果存储在缓存中,当再次出现相同的输入时,可以提供缓存的结果,从而节省了计算时间。lru_cache函数,当作为装饰器应用时,允许一个可选的maxsize参数,maxsize参数决定了缓存的最大大小(即,它为多少个不同的输入值存储结果)。如果maxsize参数设置为none,则禁用lru特性,缓存可以不受约束地增长,这会消耗很多的内存。这是最简单的空间换时间的优化方法。
10、向量化
import numpy as np  def test_11_v0(n):   # baseline version   # (inefficient way of summing numbers in a range)   output = 0   for i in range(0, n):     output = output + i    return output  def test_11_v1(n):   # improved version   # (# efficient way of summing numbers in a range)   output = np.sum(np.arange(n))   return output  
向量化一般用于机器学习的数据处理库numpy和pandas
# summary of test results      baseline: 32.936 ns per loop      improved: 1.171 ns per loop % improvement: 96.4 %      speedup: 28.13x  
11、避免创建中间列表
使用filterfalse可以避免创建中间列表。它有助于使用更少的内存。
def test_12_v0(numbers):   # baseline version (inefficient way)   filtered_data = []   for i in numbers:     filtered_data.extend(list(         filter(lambda x: x % 5 == 0,                 range(1, i**2))))      return filtered_data  
使用python的内置itertools的filterfalse函数实现相同功能的改进版本。
from itertools import filterfalse  def test_12_v1(numbers):   # improved version   # (using filterfalse)   filtered_data = []   for i in numbers:     filtered_data.extend(list(         filterfalse(lambda x: x % 5 != 0,                     range(1, i**2))))          return filtered_data  
这个方法根据用例的不同,执行速度可能没有显著提高,但通过避免创建中间列表可以降低内存使用。我们这里获得了131倍的提高
# summary of test results      baseline: 333167.790 ns per loop      improved: 2541.850 ns per loop % improvement: 99.2 %      speedup: 131.07x  
12、高效连接字符串
任何使用+操作符的字符串连接操作都会很慢,并且会消耗更多内存。使用join代替。
def test_13_v0(l_strings):   # baseline version (inefficient way)   # (concatenation using the += operator)   output =    for a_str in l_strings:     output += a_str    return output  def test_13_v1(numbers):   # improved version   # (using join)   output_list = []   for a_str in l_strings:     output_list.append(a_str)    return .join(output_list)  
该测试需要一种简单的方法来生成一个较大的字符串列表,所以写了一个简单的辅助函数来生成运行测试所需的字符串列表。
from faker import faker  def generate_fake_names(count : int=10000):   # helper function used to generate a   # large-ish list of names   fake = faker()   output_list = []   for _ in range(count):     output_list.append(fake.name())    return output_list  l_strings = generate_fake_names(count=50000)  
结果如下:
# summary of test results      baseline: 32.423 ns per loop      improved: 21.051 ns per loop % improvement: 35.1 %      speedup: 1.54x  
使用连接函数而不是使用+运算符加速1.5倍。为什么连接函数更快?
使用+操作符的字符串连接操作的时间复杂度为o(n²),而使用join函数的字符串连接操作的时间复杂度为o(n)。
总结
本文介绍了一些简单的方法,将python for循环的提升了1.3到970x。
使用python内置的map()函数代替显式的for循环加速970x
使用set代替嵌套的for循环加速498x[技巧#3]
使用itertools的filterfalse函数加速131x
使用lru_cache函数使用memoization加速57x


苹果唯一敌手:三星机海战术“围剿苹果” 力压HTC
2020爱采购行业峰会-电子行业专场 即将开幕
一加2跑分评测 5.8万跑分性能登顶
焊锡膏的模板印制
屏蔽网线和非屏蔽网线的定义
加速Python for循环的12种方法
华为P50工程样机真机曝光,曲面屏+左上角单挖孔
AdaSky宣布已经从汽车零部件制造商Sungwoo Hitech获得了新一轮2000万美元的融资
研究人员发现苹果的Macbook中的T2安全芯片存在已无法修复的缺陷
“云朵”智能问询机器人亮相,颜值与才艺兼备
沃尔沃XC40 纯电版举行车主交车仪式
鸿蒙OS 线程管理开发指导
如何选择PCB基材?
模拟滤波器
瑞萨电子推出具有AI功能的可扩展人机界面SMARC SoM解决方案
智能驾驶已取得空前的进步_核心技术还在不断完善
基于单片机和传感器的鸡舍温度监控系统设计
VueAdminWork—2.0
如何让锂离子改变沉积形态并提高锂的循环稳定性
半导体激光器应用