摘要:最近了解了提供的一個(gè)外部函數(shù)庫(kù)它提供了語(yǔ)言兼容的幾種數(shù)據(jù)類(lèi)型,并且可以允許調(diào)用編譯好的庫(kù)。這里是閱讀相關(guān)資料的一個(gè)記錄,內(nèi)容大部分來(lái)自官方文檔。注意,提供的接口會(huì)在不同系統(tǒng)上有出入,比如為了加載動(dòng)態(tài)鏈接庫(kù),在上提供的是而在上提供的是和。
參考資料
https://docs.python.org/2.7/l...
http://www.ibm.com/developerw...
ctypes簡(jiǎn)介一直對(duì)不同語(yǔ)言間的交互感興趣,python和C語(yǔ)言又深有淵源,所以對(duì)python和c語(yǔ)言交互產(chǎn)生了興趣。
最近了解了python提供的一個(gè)外部函數(shù)庫(kù) ctypes, 它提供了C語(yǔ)言兼容的幾種數(shù)據(jù)類(lèi)型,并且可以允許調(diào)用C編譯好的庫(kù)。
這里是閱讀相關(guān)資料的一個(gè)記錄,內(nèi)容大部分來(lái)自官方文檔。
ctypes 提供了一些原始的C語(yǔ)言兼容的數(shù)據(jù)類(lèi)型,參見(jiàn)下表,其中第一列是在ctypes庫(kù)中定義的變量類(lèi)型,第二列是C語(yǔ)言定義的變量類(lèi)型,第三列是Python語(yǔ)言在不使用ctypes時(shí)定義的變量類(lèi)型。
| ctypes type | C type | Python type | |--------------+----------------------------------------+----------------------------| | c_bool | _Bool | bool (1) | | c_char | char | 1-character string | | c_wchar | wchar_t | 1-character unicode string | | c_byte | char | int/long | | c_ubyte | unsigned char | int/long | | c_short | short | int/long | | c_ushort | unsigned short | int/long | | c_int | int | int/long | | c_uint | unsigned int | int/long | | c_long | long | int/long | | c_ulong | unsigned long | int/long | | c_longlong | __int64 or long long | int/long | | c_ulonglong | unsigned __int64 or unsigned long long | int/long | | c_float | float | float | | c_double | double | float | | c_longdouble | long double | float | | c_char_p | char * (NUL terminated) | string or None | | c_wchar_p | wchar_t * (NUL terminated) | unicode or None | | c_void_p | void * | int/long or None |
創(chuàng)建簡(jiǎn)單的ctypes類(lèi)型如下:
>>> c_int() c_long(0) >>> c_char_p("Hello, World") c_char_p("Hello, World") >>> c_ushort(-3) c_ushort(65533) >>>
使用 .value 訪問(wèn)和改變值:
>>> i = c_int(42) >>> print i c_long(42) >>> print i.value 42 >>> i.value = -99 >>> print i.value -99 >>>
改變指針類(lèi)型的變量值:
>>> s = "Hello, World" >>> c_s = c_char_p(s) >>> print c_s c_char_p("Hello, World") >>> c_s.value = "Hi, there" >>> print c_s c_char_p("Hi, there") >>> print s # 一開(kāi)始賦值的字符串并不會(huì)改變, 因?yàn)檫@里指針的實(shí)例改變的是指向的內(nèi)存地址,不是直接改變內(nèi)存里的內(nèi)容 Hello, World >>>
如果需要直接操作內(nèi)存地址的數(shù)據(jù)類(lèi)型:
>>> from ctypes import * >>> p = create_string_buffer(3) # create a 3 byte buffer, initialized to NUL bytes >>> print sizeof(p), repr(p.raw) 3 "x00x00x00" >>> p = create_string_buffer("Hello") # create a buffer containing a NUL terminated string >>> print sizeof(p), repr(p.raw) # .raw 訪問(wèn)內(nèi)存里存儲(chǔ)的內(nèi)容 6 "Hellox00" >>> print repr(p.value) # .value 訪問(wèn)值 "Hello" >>> p = create_string_buffer("Hello", 10) # create a 10 byte buffer >>> print sizeof(p), repr(p.raw) 10 "Hellox00x00x00x00x00" >>> p.value = "Hi" >>> print sizeof(p), repr(p.raw) 10 "Hix00lox00x00x00x00x00" >>>
下面的例子演示了使用C的數(shù)組和結(jié)構(gòu)體:
>>> class POINT(Structure): # 定義一個(gè)結(jié)構(gòu),內(nèi)含兩個(gè)成員變量 x,y,均為 int 型 ... _fields_ = [("x", c_int), ... ("y", c_int)] ... >>> point = POINT(2,5) # 定義一個(gè) POINT 類(lèi)型的變量,初始值為 x=2, y=5 >>> print point.x, point.y # 打印變量 2 5 >>> point = POINT(y=5) # 重新定義一個(gè) POINT 類(lèi)型變量,x 取默認(rèn)值 >>> print point.x, point.y # 打印變量 0 5 >>> POINT_ARRAY = POINT * 3 # 定義 POINT_ARRAY 為 POINT 的數(shù)組類(lèi)型 # 定義一個(gè) POINT 數(shù)組,內(nèi)含三個(gè) POINT 變量 >>> pa = POINT_ARRAY(POINT(7, 7), POINT(8, 8), POINT(9, 9)) >>> for p in pa: print p.x, p.y # 打印 POINT 數(shù)組中每個(gè)成員的值 ... 7 7 8 8 9 9
創(chuàng)建指針實(shí)例
>>> from ctypes import * >>> i = c_int(42) >>> pi = pointer(i) >>> >>> pi.contents c_long(42) >>>
使用cast()類(lèi)型轉(zhuǎn)換
>>> class Bar(Structure): ... _fields_ = [("count", c_int), ("values", POINTER(c_int))] ... >>> bar = Bar() >>> bar.values = (c_int * 3)(1, 2, 3) >>> bar.count = 3 >>> for i in range(bar.count): ... print bar.values[i] ... 1 2 3 >>> >>> bar = Bar() >>> bar.values = cast((c_byte * 4)(), POINTER(c_int)) # 這里轉(zhuǎn)成需要的類(lèi)型 >>> print bar.values[0] 0 >>>
類(lèi)似于C語(yǔ)言定義函數(shù)時(shí),會(huì)先定義返回類(lèi)型,然后具體實(shí)現(xiàn)再定義,當(dāng)遇到下面這種情況時(shí),也需要這么干:
>>> class cell(Structure): ... _fields_ = [("name", c_char_p), ... ("next", POINTER(cell))] ... Traceback (most recent call last): File "調(diào)用.so/.dll", line 1, in ? File " ", line 2, in cell NameError: name "cell" is not defined >>> # 不能調(diào)用自己,所以得像下面這樣 >>> from ctypes import * >>> class cell(Structure): ... pass ... >>> cell._fields_ = [("name", c_char_p), ... ("next", POINTER(cell))] >>>
可以簡(jiǎn)單地將"so"和"dll"理解成Linux和windows上動(dòng)態(tài)鏈接庫(kù)的指代,這里我們以Linux為例。注意,ctypes提供的接口會(huì)在不同系統(tǒng)上有出入,比如為了加載動(dòng)態(tài)鏈接庫(kù), 在Linux上提供的是 cdll, 而在Windows上提供的是 windll 和 oledll 。
加載動(dòng)態(tài)鏈接庫(kù)from ctypes import * >>> cdll.LoadLibrary("libc.so.6")調(diào)用加載的函數(shù)>>> libc = CDLL("libc.so.6") >>> libc >>>
>>> print libc.time(None) 1150640792 >>> print hex(windll.kernel32.GetModuleHandleA(None)) 0x1d000000 >>>設(shè)置個(gè)性化參數(shù)
ctypes會(huì)尋找 _as_paramter_ 屬性來(lái)用作調(diào)用函數(shù)的參數(shù)傳入,這樣就可以傳入自己定義的類(lèi)作為參數(shù),示例如下:
>>> class Bottles(object): ... def __init__(self, number): ... self._as_parameter_ = number ... >>> bottles = Bottles(42) >>> printf("%d bottles of beer ", bottles) 42 bottles of beer 19 >>>指定函數(shù)需要參數(shù)類(lèi)型和返回類(lèi)型
用 argtypes 和 restype 來(lái)指定調(diào)用的函數(shù)返回類(lèi)型。
>>> printf.argtypes = [c_char_p, c_char_p, c_int, c_double] >>> printf("String "%s", Int %d, Double %f ", "Hi", 10, 2.2) String "Hi", Int 10, Double 2.200000 37 >>> >>> strchr = libc.strchr >>> strchr("abcdef", ord("d")) 8059983 >>> strchr.restype = c_char_p # c_char_p is a pointer to a string >>> strchr("abcdef", ord("d")) "def" >>> print strchr("abcdef", ord("x")) None >>>
這里我只是列出了 ctypes 最基礎(chǔ)的部分,還有很多細(xì)節(jié)請(qǐng)參考官方文檔。
題外話這兩天文章沒(méi)有寫(xiě),先是早出晚歸出去玩了一整天,然后加班到凌晨3點(diǎn)左右,一天一篇計(jì)劃劃水得嚴(yán)重啊…
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.hztianpu.com/yun/44326.html
摘要:調(diào)用以回調(diào)函數(shù)地址為參數(shù)的函數(shù)這個(gè)主題就稍微繞一些了,也就是說(shuō)在接口中,需要傳入回調(diào)函數(shù)作為參數(shù)。這個(gè)問(wèn)題在中也可以解決,并且回調(diào)函數(shù)可以用定義。代碼代碼很簡(jiǎn)單回調(diào)函數(shù)的傳入?yún)?shù)為,返回參數(shù)也是。 項(xiàng)目中要對(duì)一個(gè)用 C 編寫(xiě)的 .so 庫(kù)進(jìn)行邏輯自測(cè)。這項(xiàng)工作,考慮到靈活性,我首先考慮用 Python 來(lái)完成。 研究了一些資料,采用 python 的 ctypes 來(lái)完成這項(xiàng)工作。已經(jīng)...
摘要:科大訊飛的語(yǔ)音識(shí)別客戶端提供了下的語(yǔ)言,卻沒(méi)有的。有了,我們就可以很輕松的用來(lái)使用科大訊飛的語(yǔ)音識(shí)別了。通過(guò)函數(shù)加載動(dòng)態(tài)庫(kù)在里面調(diào)用函數(shù)時(shí)主要是注意參數(shù)的類(lèi)型。 ctypes 是 Python 的一個(gè)模塊,它提供了C語(yǔ)言相關(guān)的數(shù)據(jù)類(lèi)型,可以方便的調(diào)用C語(yǔ)言編寫(xiě)的DLL(Windows 動(dòng)態(tài)鏈接庫(kù))和so(Linux動(dòng)態(tài)鏈接庫(kù))??拼笥嶏w的語(yǔ)音識(shí)別客戶端SDK提供了Linux下的C語(yǔ)言S...
摘要:上篇文章我許了一個(gè)愿,就是想讓大家多多關(guān)注我,然后我的粉絲就蹭蹭的漲了好幾百,謝謝大家的厚愛(ài)??墒俏野l(fā)現(xiàn)粉絲是漲了,三連變少了,謝謝大家這次給我三連,我一定再接再厲。地址的尋找陽(yáng)光總值,種植一個(gè)豌豆需要,非常不夠用。 目錄 前言 游戲的安裝 思路 ? ? ? 一句話總結(jié) ? ? ? 大概的思...
摘要:?jiǎn)栴}近日在做一組聲紋聚類(lèi)時(shí),使用了另一團(tuán)隊(duì)同學(xué)開(kāi)發(fā)的聲紋距離算法。該算法對(duì)外提供的是一組包,需要使用方自己去使用。本次使用過(guò)程中,就遇到傳參的問(wèn)題。修改后代碼如下結(jié)論中傳遞指針類(lèi)型參數(shù)需要顯式聲明函數(shù)的參數(shù),返回類(lèi)型。 問(wèn)題 近日在做一組聲紋聚類(lèi)時(shí),使用了另一團(tuán)隊(duì)同學(xué)開(kāi)發(fā)的聲紋距離算法。該算法對(duì)外提供的是一組so包,需要使用方自己去使用。在python中調(diào)用純so包一般使用ctypes...
Centos的版本還是比較的多的,那么,我們要怎么在這個(gè)服務(wù)器上,去進(jìn)行安裝python3呢,具體的方法是什么呢?如果安裝遇到問(wèn)題的話,該怎么去進(jìn)行解決呢?下面就給詳細(xì)解答下?! ∫弧entOS安裝Python3 centos7自帶版本是python2.7 如果要用的3.0以上的版本需要手動(dòng)安裝,下載地址:https://www.python.org/ftp/python/ 1、先查看...
閱讀 2631·2023-04-26 01:44
閱讀 2684·2021-09-10 10:50
閱讀 1496·2019-08-30 15:56
閱讀 2440·2019-08-30 15:44
閱讀 575·2019-08-29 11:14
閱讀 3486·2019-08-26 11:56
閱讀 3086·2019-08-26 11:52
閱讀 988·2019-08-26 10:27