轉(zhuǎn)自:http://www.byhy.net/tut/py/etc/calldll/
為什么需要調(diào)用動態(tài)鏈接庫
有大量優(yōu)秀的庫聚唐,直接以本地庫(機器代碼)提供的,這些庫基本都是c/c++語言 編寫的舍悯。
通常要調(diào)用它們贝奇,需要使用 c/c++ 語言虹菲。
如果你的應(yīng)用,已經(jīng)用python開發(fā)了掉瞳,能不能用python直接調(diào)用這些庫毕源,使用這些庫的功能呢?
當(dāng)然可以陕习。 python語言的官方解釋器 CPython 本身就是用C語言開發(fā)的霎褐,可以很方便的從python代碼中調(diào)用動態(tài)鏈接庫。
動態(tài)鏈接庫该镣,在不同的操作系統(tǒng)(Windows冻璃、Linux),文件格式不同,但是python調(diào)用它們的 方法都差不多省艳。
我們這里以Windows平臺的dll為例娘纷。
參考 https://docs.python.org/3/library/ctypes.html
一個例子
點擊這里下載 http://cdn1.python3.vip/files/py/etc/Dll1.dll
動態(tài)鏈接庫中有這樣的函數(shù)
void SayHello()
{
MessageBox(NULL,
TEXT("白月黑羽向您問好~~~"),
TEXT("白月黑羽培訓(xùn)專用"),
MB_OK);
}
使用如下代碼加載dll ,并且調(diào)用函數(shù) SayHello
from ctypes import *
# Load DLL into memory.
lib = CDLL ("e:\\Dll1.dll")
lib.SayHello()
有參數(shù)
from ctypes import *
# Load DLL into memory.
lib = CDLL ("e:\\Dll1.dll")
lib.StrAppend('白月黑羽')
ret = lib.IntAdd(3,4)
print(ret)
通過python可以很方便的使用不同參數(shù)調(diào)用跋炕,無須編譯運行赖晶。
如果是測試,甚至比直接用c語言寫測試代碼方便辐烂。
上面的 IntAdd 和 StrAppend遏插, 我們直接使用了Python對象作為參數(shù)傳入。
實際上纠修,底層的ctypes庫調(diào)用c語言庫胳嘲,不能直接傳遞python對象的,需要轉(zhuǎn)化為c語言接口對應(yīng)的類型扣草。
它會根據(jù)我們使用的python對象類型了牛,猜測應(yīng)該轉(zhuǎn)化為什么類型的數(shù)據(jù)。
Python數(shù)據(jù)對象和C語言的數(shù)據(jù)對象的對應(yīng)關(guān)系見 https://docs.python.org/3/library/ctypes.html#fundamental-data-types
特別要注意 w_char 和 char 的區(qū)別德召,前者對應(yīng)python 3中的 字符串白魂, 后者對應(yīng) 字節(jié)串
指定參數(shù)類型
前面的示例,我們直接使用了Python對象作為參數(shù)傳入上岗。
好像也沒有什么問題福荸。
但是建議大家最好還是直接告訴 ctypes 你的參數(shù)類型。
特別是從python 對象類型 不能唯一對應(yīng)c語言類型的肴掷。比如 char short int long 這些敬锐。
否則可能帶來意想不到的問題。參考 https://stackoverflow.com/questions/24377845/ctype-why-specify-argtypes
怎么告訴呢呆瞻?
from ctypes import *
# Load DLL into memory.
lib = CDLL ("e:\\Dll1.dll")
lib.StrAppend.argtypes = [c_wchar_p]
lib.StrAppend('白月黑羽')
也可以
lib.StrAppend(c_wchar_p('白月黑羽'))
指定返回值類型
看看下面的例子
from ctypes import *
lib = CDLL ("e:\\Dll1.dll")
ret = lib.IntAdd(3,4)
print(ret)
ret2 = lib.StrAppend('白月黑羽')
print(ret2)
第1個是對的台夺,第2個怎么是數(shù)字?
返回類型痴脾,ctypes 缺省認為是 C int 類型
不是int 類型的颤介,可以通過 function 對象的restype 屬性指定
from ctypes import *
# Load DLL into memory.
lib = CDLL ("e:\\Dll1.dll")
ret = lib.IntAdd(3,4)
print(ret)
lib.StrAppend.restype = c_wchar_p
ret2 = lib.StrAppend('白月黑羽')
print(ret2)
lib.BytesAppend.restype = c_char_p
ret3 = lib.BytesAppend(b'\x39\x99')
print(ret3)
復(fù)合類型參數(shù)
參考
導(dǎo)出 c++ 函數(shù)
由于 c++ 對符號名的魔改,c++ 語法編譯的函數(shù)赞赖,直接ctypes調(diào)用會發(fā)現(xiàn)找不到函數(shù)
要調(diào)用怎么辦滚朵?
一種方法是加上 extern "C"
的說明,重新編譯動態(tài)鏈接庫前域。
extern "C" __declspec(dllexport) void SayHello();
如果沒有源代碼辕近,可以通過工具(比如 dllexp )找到 這個dll 導(dǎo)出的 c++ 修改后函數(shù)名字,然后調(diào)用它
參考 https://stackoverflow.com/questions/21184911/c-dll-called-from-python