詳解如何利用Cython為Python代碼加速
引言
通常,在 Python 中寫(xiě)循環(huán)(特別是多重循環(huán))非常的慢,在文章 http://www.dhdzp.com/article/133807.htm中,我們的元胞自動(dòng)機(jī)的狀態(tài)更新函數(shù) update_state 使用了兩重循環(huán),所以我們嘗試用 Cython 重構(gòu)該方法。
代碼
我們?cè)谕募A下新建一個(gè) update.pyx 文件,寫(xiě)入如下內(nèi)容
import numpy as np
cimport numpy as np
cimport cython
DTYPE = np.float
ctypedef np.float_t DTYPE_t
def update_state(np.ndarray[DTYPE_t, ndim=2] cells):
return update_state_c(cells)
@cython.boundscheck(False)
@cython.wraparound(False)
cdef np.ndarray[DTYPE_t, ndim=2] update_state_c(np.ndarray[DTYPE_t, ndim=2] cells):
"""更新一次狀態(tài)"""
cdef unsigned int i
cdef unsigned int j
cdef np.ndarray[DTYPE_t, ndim=2] buf = np.zeros((cells.shape[0], cells.shape[1]), dtype=DTYPE)
cdef DTYPE_t neighbor_num
for i in range(1, cells.shape[0] - 1):
for j in range(1, cells.shape[0] - 1):
# 計(jì)算該細(xì)胞周?chē)拇婊罴?xì)胞數(shù)
neighbor_num = cells[i, j-1] + cells[i, j+1] + cells[i+1, j] + cells[i-1, j] +\
cells[i-1, j-1] + cells[i-1, j+1] +\
cells[i+1, j-1] + cells[i+1, j+1]
if neighbor_num == 3:
buf[i, j] = 1
elif neighbor_num == 2:
buf[i, j] = cells[i, j]
else:
buf[i, j] = 0
return buf
update_state_c 函數(shù)上的兩個(gè)裝飾器是用來(lái)關(guān)閉 Cython 的邊界檢查的。
在同文件下新建一個(gè) setup.py 文件
import numpy as np
from distutils.core import setup
from Cython.Build import cythonize
setup(
name="Cython Update State",
ext_modules=cythonize("update.pyx"),
include_dirs=[np.get_include()]
)
因?yàn)樵?Cython 文件中使用了 NumPy 的頭文件,所以我們需要在 setup.py 將其包含進(jìn)去。
執(zhí)行 python setup.py build_ext --inplace 后,同文件夾下會(huì)生成一個(gè) update.cp36-win_amd64.pyd 的文件,這就是編譯好的 C 擴(kuò)展。
我們修改原始的代碼,首先在文件頭部加入 import update as cupdate,然后修改更新方法如下
def update_state(self): """更新一次狀態(tài)""" self.cells = cupdate.update_state(self.cells) self.timer += 1
將原方法名就改為 update_state_py 即可,運(yùn)行腳本,無(wú)異常。
測(cè)速
我們編寫(xiě)一個(gè)方法來(lái)測(cè)試一下使用 Cython 可以帶來(lái)多少速度的提升
def test_time():
import time
game = GameOfLife(cells_shape=(60, 60))
t1 = time.time()
for _ in range(300):
game.update_state()
t2 = time.time()
print("Cython Use Time:", t2 - t1)
del game
game = GameOfLife(cells_shape=(60, 60))
t1 = time.time()
for _ in range(300):
game.update_state_py()
t2 = time.time()
print("Native Python Use Time:", t2 - t1)
運(yùn)行該方法,在我的電腦上輸出如下
Cython Use Time: 0.007000446319580078
Native Python Use Time: 4.342248439788818
速度提升了 600 多倍。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
深度學(xué)習(xí)TextLSTM的tensorflow1.14實(shí)現(xiàn)示例
這篇文章主要為大家介紹了深度學(xué)習(xí)TextLSTM的tensorflow1.14實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
python 實(shí)現(xiàn)讓字典的value 成為列表
今天小編就為大家分享一篇python 實(shí)現(xiàn)讓字典的value 成為列表,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-12-12
Python列表推導(dǎo)式實(shí)現(xiàn)代碼實(shí)例
這篇文章主要介紹了Python列表推導(dǎo)式實(shí)現(xiàn)代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-09-09
如何使用python轉(zhuǎn)移mysql數(shù)據(jù)庫(kù)中的全部數(shù)據(jù)
這篇文章主要為大家詳細(xì)介紹了如何使用python轉(zhuǎn)移mysql數(shù)據(jù)庫(kù)中的全部數(shù)據(jù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解下2024-11-11
python3 flask實(shí)現(xiàn)文件上傳功能
這篇文章主要為大家詳細(xì)介紹了python3 flask實(shí)現(xiàn)文件上傳功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-08-08
TensorFlow實(shí)現(xiàn)簡(jiǎn)單線性回歸
這篇文章主要為大家詳細(xì)介紹了TensorFlow實(shí)現(xiàn)簡(jiǎn)單線性回歸,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
python 讀取dicom文件,生成info.txt和raw文件的方法
今天小編就為大家分享一篇python 讀取dicom文件,生成info.txt和raw文件的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-01-01

