解释与Python中的动态调度惩罚

问题描述 投票:4回答:1

我看了Brandon Rhodes关于Cython的演讲 - “EXE的日子在我们身上”。

布兰登在09:30提到,对于一段特定的短代码,跳过解释给出了40%的加速,而跳过分配和发送则给出了574%的加速(10:10)。

我的问题是 - 如何测量特定的代码?是否需要手动提取底层c命令,然后以某种方式使运行时运行它们?

这是一个非常有趣的观察,但我如何重新创建实验呢?

python dynamic-programming cython performance-testing dispatch
1个回答
6
投票

我们来看看这个python函数:

def py_fun(i,N,step):
     res=0.0
     while i<N:
        res+=i
        i+=step
     return res

并使用ipython-magic来计时:

In [11]: %timeit py_fun(0.0,1.0e5,1.0)
10 loops, best of 3: 25.4 ms per loop

解释器将运行生成的字节码并对其进行解释。但是,我们可以通过使用cython for / cython化相同的代码来删除解释器:

%load_ext Cython
%%cython
def cy_fun(i,N,step):
     res=0.0
     while i<N:
        res+=i
        i+=step
     return res

我们的速度提高了50%:

In [13]: %timeit cy_fun(0.0,1.0e5,1.0)
100 loops, best of 3: 10.9 ms per loop

当我们查看生成的c代码时,我们看到正确的函数被直接调用,而不需要解释/调用ceval,这是在删除样板代码之后:

static PyObject *__pyx_pf_4test_cy_fun(CYTHON_UNUSED PyObject *__pyx_self, PyObject *__pyx_v_i, PyObject *__pyx_v_N, PyObject *__pyx_v_step) {
  ...
  while (1) {
    __pyx_t_1 = PyObject_RichCompare(__pyx_v_i, __pyx_v_N, Py_LT); 
    ...
    __pyx_t_2 = __Pyx_PyObject_IsTrue(__pyx_t_1);
    ...
    if (!__pyx_t_2) break;
    ...
    __pyx_t_1 = PyNumber_InPlaceAdd(__pyx_v_res, __pyx_v_i);
    ...
    __pyx_t_1 = PyNumber_InPlaceAdd(__pyx_v_i, __pyx_v_step); 
  }
  ...
  return __pyx_r;
}

但是,这个cython函数处理python-objects而不是c-style float,所以在函数PyNumber_InPlaceAdd中有必要弄清楚这些对象(整数,浮点数,其他东西?)到底是什么,并将此调用分配给正确的函数哪个会做的。

在cython的帮助下,我们还可以消除对此调度的需要,并直接调用浮点数的乘法:

 %%cython
 def c_fun(double i,double N, double step):
      cdef double res=0.0
      while i<N:
         res+=i
         i+=step
      return res

在这个版本中,iNstepres是c风格的双打而不再是python对象。所以不再需要调用像PyNumber_InPlaceAdd这样的调度函数,但是我们可以直接调用+-operator来获取double

static PyObject *__pyx_pf_4test_c_fun(CYTHON_UNUSED PyObject *__pyx_self, double __pyx_v_i, double __pyx_v_N, double __pyx_v_step) {
  ...
  __pyx_v_res = 0.0;  
  ... 
  while (1) {
    __pyx_t_1 = ((__pyx_v_i < __pyx_v_N) != 0);
    if (!__pyx_t_1) break;
    __pyx_v_res = (__pyx_v_res + __pyx_v_i);
    __pyx_v_i = (__pyx_v_i + __pyx_v_step);
  }
  ...
  return __pyx_r;
}

结果是:

In [15]: %timeit c_fun(0.0,1.0e5,1.0)
10000 loops, best of 3: 148 µs per loop

现在,与没有解释器但是有调度的版本相比,这是几乎100的加速。

实际上,要说,调度+分配是这里的瓶颈(因为消除它导致几乎100倍的加速)是一个谬误:解释器负责超过50%的运行时间(15毫秒)和发送和分配“仅”10ms。


但是,对于性能而言,“解释器”和动态调度存在的问题更多:浮点数是不可变的,因此每次更改时,必须在垃圾收集器中创建和注册/取消注册新对象。

我们可以引入可变浮点数,这些浮点数已就地更改,不需要注册/取消注册:

%%cython
cdef class MutableFloat: 
 cdef double x      
 def __cinit__(self, x):
    self.x=x         
 def __iadd__(self, MutableFloat other):
    self.x=self.x+other.x
    return self
 def __lt__(MutableFloat self,  MutableFloat other):
    return self.x<other.x
 def __gt__(MutableFloat self, MutableFloat other):
    return self.x>other.x
 def __repr__(self):
    return str(self.x)

时间(现在我使用不同的机器,所以时间有点不同):

def py_fun(i,N,step,acc):
        while i<N:
             acc+=i
             i+=step
        return acc

%timeit py_fun(1.0, 5e5,1.0,0.0)
30.2 ms ± 1.12 ms per loop (mean ± std. dev. of 7 runs, 10 loops each 
%timeit cy_fun(1.0, 5e5,1.0,0.0)
16.9 ms ± 612 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)
%timeit i,N,step,acc=MutableFloat(1.0),MutableFloat(5e5),MutableFloat(1
    ...: .0),MutableFloat(0.0); py_fun(i,N,step,acc)
23 ms ± 254 µs per loop (mean ± std. dev. of 7 runs, 10 loops each)
%timeit i,N,step,acc=MutableFloat(1.0),MutableFloat(5e5),MutableFloat(1
...: .0),MutableFloat(0.0); cy_fun(i,N,step,acc)
11 ms ± 66.2 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)

不要忘记重新初始化i,因为它是可变的!结果

            immutable       mutable
 py_fun       30ms           23ms
 cy_fun       17ms           11ms

因此,在使用解释器的版本中注册/取消注册浮动(我不确定没有其他任何东西发挥作用)需要多达7ms(约20%),而在没有解释器的版本中需要多达33%。

现在看起来:

  • 口译员使用40%(13/30)的时间
  • 高达33%的时间用于动态调度
  • 最多20%的时间用于创建/删除临时对象
  • 算术运算约1%

另一个问题是数据的局部性,这对于存储器带宽限制问题变得明显:如果数据线性地连续处理一个连续的存储器地址,则现代缓存很有效。这适用于循环std::vector<>(或array.array),但不适用于循环python列表,因为此列表包含指向内存中任何位置的指针。

考虑以下python脚本:

#list.py
N=int(1e7)
lst=[0]*int(N)
for i in range(N):
  lst[i]=i
print(sum(lst)) 

#byte
N=int(1e7)
b=bytearray(8*N)
m=memoryview(b).cast('L') #reinterpret as an array of unsigned longs
for i in range(N):
  m[i]=i
print(sum(m))

它们都创建了1e7整数,第一个版本是Python整数,第二个是低级c-int,它们被连续放置在内存中。

有趣的是,这些脚本产生了多少缓存未命中(D):

valgrind --tool=cachegrind python list.py 
...
D1  misses:        33,964,276  (   27,473,138 rd   +     6,491,138 wr)

valgrind --tool=cachegrind python bytearray.py 
...
D1  misses:         4,796,626  (    2,140,357 rd   +     2,656,269 wr)

这意味着python-integers有8次缓存未命中。它的一部分原因是,python整数需要超过8个字节(可能是32字节,即因子4)的内存,并且(可能不是100%肯定,因为相邻的整数是在彼此之后创建的,所以机会很高,它们存储在内存中的某个地方,需要进一步调查),因为事实上它们没有在内存中对齐,因为bytearray的c整数就是这种情况。

© www.soinside.com 2019 - 2024. All rights reserved.