Python中ctypes的使用整理

转载:点击打开链接

ctypesPython的一个外部库,提供和C语言兼容的数据类型,可以很方便地调用C DLL中的函数。ctypes的官方文档在这里

1. ctypes基本数据类型映射表

参数类型预先设定好,或者在调用函数时再把参数转成相应的c_***类型。ctypes的类型对应如下:

ctypes typeC typePython Type
c_charchar1-character string
c_wcharwchar_t1-character unicode string
c_bytecharint/long
c_ubyteunsigned charint/long
c_boolboolbool
c_shortshortint/long
c_ushortunsigned shortint/long
c_intintint/long
c_uintunsigned intint/long
c_longlongint/long
c_ulongunsigned longint/long
c_longlong__int64 or longlongint/long
c_ulonglongunsigned __int64 or unsigned long longint/long
c_floatfloatfloat
c_doubledoublefloat
c_longdoublelong double floatfloat
c_char_pchar *string or None
c_wchar_pwchar_t *unicode or None
c_void_pvoid *int/long or None
   

对应的指针类型是在后面加上"_p",如int*是c_int_p等等。在python中要实现c语言中的结构,需要用到类。 

2. 加载DLL

访问dll,首先需引入ctypes库 
from ctypes import *   

假设你已经有了一个的DLL(名字是add.dll),且该DLL有一个符合cdecl(这里强调调用约定是因为,stdcall调用约定和cdecl调用约定声明的导出函数,在使用python加载时使用的加载函数是不同的,后面会有说明)调用约定的导出函数Add。
stdcall调用约定:两种加载方式

[python]  view plain  copy
  1. Objdll = ctypes.windll.LoadLibrary("dllpath")    
  2. Objdll = ctypes.WinDLL("dllpath")     

cdecl调用约定:也有两种加载方式

[python]  view plain  copy
  1. Objdll = ctypes.cdll.LoadLibrary("dllpath")    
  2. Objdll = ctypes.CDLL("dllpath")    

其实windll和cdll分别是WinDLL类和CDll类的对象。 

 

3. 调用DLL方法

加载dll后会返回一个DLL对象,使用其中的函数方法则相当于操作该对象的对应属性。

注意,经过stdcall声明的方法,如果不是用def文件声明的导出函数或者extern “C” 声明的话,编译器会对函数名进行修改

函数参数申明,通过设置函数的argtypes属性

函数返回类型,函数默认返回c_int类型,如果需要返回其他类型,需要设置函数的restype属性

 

4. 指针与引用

常用的通过调用ctypes类型的指针函数来创建指针实例:

[python]  view plain  copy
  1. from ctype import *  
  2. i = c_int(1)  
  3. pi = POINTER(i)  

对指针实例赋值只会改变其指向的内存地址,而不是改变内存的内容,与其他语言类似,如需要可改变内容的字符串,可须使用create_string_buffer()

[python]  view plain  copy
  1. >>> p = create_string_buffer("Hello"10)  # create a 10 byte buffer  
  2. >>> print sizeof(p), repr(p.raw)  
  3. 10 'Hello/x00/x00/x00/x00/x00'  

不带参数的调用指针类型创建一个NULL指针, NULL指针有一个False布尔值

[python]  view plain  copy
  1. >>> null_ptr = POINTER(c_int)()   
  2. >>> print bool(null_ptr)   
  3. False   

指针实例有一个contents属性,返回这个指针所指向的对象。

另外,byref()是用来传递引用参数,pointer()作为传参通常会创建一个实际的指针对象,当不需要实际指针对象时,则可使用byref()

 

5. 结构体类型处理

Structures和Unions必须继承Structure和Union基础类,它们都在ctypes模块中定义,每一个子类必须定义个_fields_属性,_fields_是一个二维的tuples列表,包含着每个field的name及type,这field类型必须是一个ctypes类型,如c_int,或者任何其他的继承ctypes的类型,如Structure, Union, Array, 指针等。

例如有一个简单结构,包含两个整型x和y,可如下初始化一个结构:

[python]  view plain  copy
  1. from ctypes import *  
  2. import types  
  3. class Test(Structure):  
  4.     _fields_ = [('x', c_int),  
  5.                 ('y', c_char)]  
  6. test1 = Test(12)  

另外,如结构体用于链表操作,即包含指向结构体指针时,若直接定义:

[python]  view plain  copy
  1. from ctypes import *  
  2. import types  
  3. class Test(Structure):  
  4.     _fields_ = [('x', c_int),  
  5.                 ('y', c_char),  
  6.                 ('next', Test)]  

则python会报错type未定义,:

[python]  view plain  copy
  1. from ctypes import *  
  2. import types  
  3. class Test(Structure):  
  4.     pass  
  5. Test._fields_ = [('x', c_int),  
  6.                 ('y', c_char),  
  7.                 ('next', POINTER(Test))]  

 

6. 数组定义

数组包含一个固定数目的相同类型的实例,常用创建数组类型是对一个数据类型与一个正数相乘,例如:

[python]  view plain  copy
  1. ArrayType = Test * 10  
[python]  view plain  copy
  1. 初始化和使用数组:  
  2.   
  3. <span style="color:#0000ff;">>>> from ctypes import *  
  4. >>> TenIntegers = c_int * 10  
  5. >>> ii = TenIntegers(12345678910)  
  6. >>> print ii  
  7. <c_long_Array_10 object at 0x...>  
  8. >>> for i in ii: print i,  
  9. </span>  

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值