Python調用ctypes使用C函數printf的方法
在Python程序中導入ctypes模塊,載入動態(tài)鏈接庫。動態(tài)鏈接庫有三種:cdll以及windows下的windll和oledll,cdll載入導出函數使用標準的cdecl調用規(guī)范的庫,而windll載入導出函數符合stdcall調用規(guī)范(Win32 API的原生約定)的庫,oledll也使用stdcall調用規(guī)范,并假設函數返回Windows的HRESULT錯誤代碼。錯誤代碼用于在出錯時自動拋出WindowsError這個Python異常,可以使用COM函數得到具體的錯誤信息。
使用cdll.msvcrt即可調用MS標準的C庫msvcrt,msvcrt包含了大部分標準C函數。
下面來看一下簡單的printf函數。
from ctypes import * msvcrt = cdll.msvcrt str = "Huanhuan!" msvcrt.printf("Hello %s\n", str)
這樣就可以使用C語言中的printf函數進行輸出。
如果在IDLE里運行的話會發(fā)現程序沒有任何輸出結果,這是因為printf是打印到真實的標準輸出,而不是sys.stdout。如果想要看到運行結果,可以在CMD里運行python test.py來查看結果,前提是已經設置好了Python的環(huán)境變量?;蛘哂幸粋€曲線方法可以在IDLE中顯示輸出結果,請曲線閱讀到文章最后。
如果使用的是Py3K,在控制臺里會看到只有開頭字符H被輸出了。因為Py3K使用的是Unicode編碼,而printf不支持該編碼,所以需要轉碼。整理出來三種改寫方法可以解決這一問題。
# A 轉為byte類型 在字符串前面加b from ctypes import * msvcrt = cdll.msvcrt str = b"Huanhuan!" msvcrt.printf(b"Hello %s\n", str) # B 使用wprintf寬字符顯示 from ctypes import * msvcrt = cdll.msvcrt str = "Huanhuan!" msvcrt.wprintf("Hello %s\n", str) # C 轉碼為utf-8 from ctypes import * msvcrt = cdll.msvcrt str = "Huanhuan!" result = "Hello " + str + "\n" result = result.encode("utf-8") msvcrt.printf(result)
最后來搞定在IDLE中曲線顯示輸出結果的方法。
from ctypes import * msvcrt = cdll.msvcrt str = b"Huanhuan!" s = create_string_buffer(100) # 必須足夠長 msvcrt.sprintf(s, b'Hello %s\n', str) print(s.value.decode('utf-8'))
先使用sprintf函數把結果輸出到s變量,然后再用Python自帶的print方法輸出s的value。
好了,通過以上的各種方法就可以解決Py3K調用C函數printf的問題了。
什么?你問我為什么費這么大勁非要用printf輸出,而不是直接用Python自帶的print?
python的print和c的printf有什么區(qū)別
print([object, ...], *, sep=' ', end='\n', file=sys.stdout, flush=False)
輸出對象到流文件,sep指定分割符,end指定結束符。參數轉換為字符串寫入輸出流,如果沒有輸出內容直接輸出end結束符。file參數必須是包含write方法的對象,默認輸出到標準輸出。
int printf( char * format, ... );
根據參數 format 字符串來轉換并格式化數據,然后將結果輸出到標準輸出設備(顯示器),直到出現字符串結束('\0')為止。
參數 format 字符串可包含下列三種字符類型:
一般文本,將會直接輸出
ASCII 控制字符,如\t、\n 等有特定含義
格式轉換字符
格式轉換為一個百分比符號(%)及其后的格式字符所組成。一般而言,每個%符號在其后都必需有一個參數與之相呼應(只有當%%轉換字符出現時會直接輸出%字符),而欲輸出的數據類型必須與其相對應的轉換字符類型相同。
相關文章
Python編程scoketServer實現多線程同步實例代碼
這篇文章主要介紹了Python編程scoketServer實現多線程同步實例代碼,小編覺得還是挺不錯的,具有一定借鑒價值,需要的朋友可以參考下2018-01-01