分享

python ctypes中文帮助文档

 dinghj 2019-04-17

   15.17。ctypes- 用于Python的外部函数库

2.5版中的新功能。

ctypes是Python的外部函数库。它提供C兼容的数据类型,并允许在DLL或共享库中调用函数。它可以用于在纯Python中包装这些库。

15.17.1。ctypes教程

注意:本教程中的代码示例doctest用于确保它们实际工作。由于某些代码示例在Linux,Windows或Mac OS X下的行为不同,因此它们在注释中包含doctest指令。

注意:某些代码示例引用了ctypes c_int类型。此类型是c_long32位系统上类型的别名。因此,c_long如果您打算如果打印,则不应该感到困惑c_int- 它们实际上是相同的类型。

   15.17.1.1。加载动态链接库

ctypes导出cdll,以及Windows windlloledll 对象,用于加载动态链接库。

您可以通过访问它们作为这些对象的属性来加载库。cdll 加载使用标准cdecl调用约定导出函数的库,而windll库使用stdcall 调用约定调用函数。oledll还使用stdcall调用约定,并假定函数返回Windows HRESULT错误代码。错误代码用于WindowsError在函数调用失败时自动引发异常。

以下是Windows的一些示例。注意,它msvcrt是包含大多数标准C函数的MS标准C库,并使用cdecl调用约定:

  1. >>> from ctypes import *
  2. >>> print windll.kernel32
  3. <WinDLL 'kernel32', handle ... at ...>
  4. >>> print cdll.msvcrt
  5. <CDLL 'msvcrt', handle ... at ...>
  6. >>> libc = cdll.msvcrt
  7. >>>

Windows会.dll自动附加常用的文件后缀。

在Linux上,需要指定包含加载库的扩展名的文件名,因此不能使用属性访问来加载库。LoadLibrary()应该使用dll加载器的 方法,或者你应该通过调用构造函数创建一个CDLL实例来加载库:

  1. >>> cdll.LoadLibrary("libc.so.6")
  2. <CDLL 'libc.so.6', handle ... at ...>
  3. >>> libc = CDLL("libc.so.6")
  4. >>> libc
  5. <CDLL 'libc.so.6', handle ... at ...>
  6. >>>

   15.17.1.2。从加载的dll访问函数

函数作为dll对象的属性进行访问:

  1. >>> from ctypes import *
  2. >>> libc.printf
  3. <_FuncPtr object at 0x...>
  4. >>> print windll.kernel32.GetModuleHandleA
  5. <_FuncPtr object at 0x...>
  6. >>> print windll.kernel32.MyOwnFunction
  7. Traceback (most recent call last):
  8. File "<stdin>", line 1, in <module>
  9. File "ctypes.py", line 239, in __getattr__
  10. func = _StdcallFuncPtr(name, self)
  11. AttributeError: function 'MyOwnFunction' not found
  12. >>>

请注意,win32系统类似于kernel32并且user32经常导出ANSI以及函数的UNICODE版本。UNICODE版本将导出W并附加到名称,而ANSI版本将导出A 并附加到名称。win32 GetModuleHandle函数返回给定模块名称的 模块句柄,具有以下C原型,并且GetModuleHandle根据是否定义了UNICODE ,使用宏来公开其中一个:

  1. /* ANSI version */
  2. HMODULE GetModuleHandleA(LPCSTR lpModuleName);
  3. /* UNICODE version */
  4. HMODULE GetModuleHandleW(LPCWSTR lpModuleName);

windll不会尝试通过魔法选择其中一个,您必须通过指定GetModuleHandleAGetModuleHandleW 显式访问所需的版本,然后分别使用字符串或unicode字符串调用它。

有时,dll导出的函数名称不是有效的Python标识符,例如"??2@YAPAXI@Z"。在这种情况下,您必须使用 getattr()检索功能:

  1. >>> getattr(cdll.msvcrt, "??2@YAPAXI@Z")
  2. <_FuncPtr object at 0x...>
  3. >>>

在Windows上,某些dll不按名称导出函数,而是按顺序导出函数。可以通过使用序号索引dll对象来访问这些函数:

  1. >>> cdll.kernel32[1]
  2. <_FuncPtr object at 0x...>
  3. >>> cdll.kernel32[0]
  4. Traceback (most recent call last):
  5. File "<stdin>", line 1, in <module>
  6. File "ctypes.py", line 310, in __getitem__
  7. func = _StdcallFuncPtr(name, self)
  8. AttributeError: function ordinal 0 not found
  9. >>>

    15.17.1.3。调用函数

您可以像调用任何其他Python一样调用这些函数。此示例使用time()函数,该函数返回自Unix纪元以来以秒为单位的系统时间,以及GetModuleHandleA()返回win32模块句柄的函数。

此示例使用NULL指针调用这两个函数(None应该用作NULL指针):

  1. >>> print libc.time(None)
  2. 1150640792
  3. >>> print hex(windll.kernel32.GetModuleHandleA(None))
  4. 0x1d000000
  5. >>>

ctypes试图保护您不要使用错误的参数数量或错误的调用约定来调用函数。不幸的是,这仅适用于Windows。它通过在函数返回后检查堆栈来完成此操作,因此虽然引发了错误,但函数被调用:

  1. >>> windll.kernel32.GetModuleHandleA()
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. ValueError: Procedure probably called with not enough arguments (4 bytes missing)
  5. >>> windll.kernel32.GetModuleHandleA(0, 0)
  6. Traceback (most recent call last):
  7. File "<stdin>", line 1, in <module>
  8. ValueError: Procedure probably called with too many arguments (4 bytes in excess)
  9. >>>

stdcall使用cdecl调用约定调用函数 时会引发相同的异常,反之亦然:

  1. >>> cdll.kernel32.GetModuleHandleA(None)
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. ValueError: Procedure probably called with not enough arguments (4 bytes missing)
  5. >>>
  6. >>> windll.msvcrt.printf("spam")
  7. Traceback (most recent call last):
  8. File "<stdin>", line 1, in <module>
  9. ValueError: Procedure probably called with too many arguments (4 bytes in excess)
  10. >>>

要找出正确的调用约定,您必须查看C头文件或要调用的函数的文档。

在Windows上,ctypes使用win32结构化异常处理来防止在使用无效参数值调用函数时因常规保护错误而崩溃:

  1. >>> windll.kernel32.GetModuleHandleA(32)
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. WindowsError: exception: access violation reading 0x00000020
  5. >>>

但是,有足够的方法可以使Python崩溃ctypes,所以无论如何你应该小心。

None,整数,长整数,字节字符串和unicode字符串是唯一可以直接用作这些函数调用中的参数的本机Python对象。 None作为C NULL指针传递,字节字符串和unicode字符串作为指针传递给包含其数据( 或)的内存块。Python整数和Python long作为平台默认C 类型传递,它们的值被屏蔽以适合C类型。char *wchar_t *int

在我们继续使用其他参数类型调用函数之前,我们必须了解有关ctypes数据类型的更多信息。

   15.17.1.4。基本数据类型

ctypes 定义了许多原始C兼容的数据类型:

ctypes类型 C型 Python类型
c_bool _Bool 布尔(1)
c_char char 1个字符的字符串
c_wchar wchar_t 1个字符的unicode字符串
c_byte char INT /长
c_ubyte unsigned char INT /长
c_short short INT /长
c_ushort unsigned short INT /长
c_int int INT /长
c_uint unsigned int INT /长
c_long long INT /长
c_ulong unsigned long INT /长
c_longlong __int64 要么 long long INT /长
c_ulonglong unsigned __int64 要么 unsigned long long INT /长
c_float float 浮动
c_double double 浮动
c_longdouble long double 浮动
c_char_p char * (NUL终止) 字符串或 None
c_wchar_p wchar_t * (NUL终止) unicode或 None
c_void_p void * int / long或 None
  1. 构造函数接受具有真值的任何对象。

所有这些类型都可以通过使用正确类型和值的可选初始化程序调用它们来创建:

  1. >>> c_int()
  2. c_long(0)
  3. >>> c_char_p("Hello, World")
  4. c_char_p('Hello, World')
  5. >>> c_ushort(-3)
  6. c_ushort(65533)
  7. >>>

由于这些类型是可变的,因此它们的值也可以在以后更改:

  1. >>> i = c_int(42)
  2. >>> print i
  3. c_long(42)
  4. >>> print i.value
  5. 42
  6. >>> i.value = -99
  7. >>> print i.value
  8. -99
  9. >>>

分配一个新的值,将指针类型的实例c_char_p, c_wchar_p以及c_void_p改变所述存储器位置它们指向,而不是内容的内存块(当然不是,因为Python字符串是不可变的):

  1. >>> s = "Hello, World"
  2. >>> c_s = c_char_p(s)
  3. >>> print c_s
  4. c_char_p('Hello, World')
  5. >>> c_s.value = "Hi, there"
  6. >>> print c_s
  7. c_char_p('Hi, there')
  8. >>> print s # first string is unchanged
  9. Hello, World
  10. >>>

但是,您应该小心,不要将它们传递给期望指向可变内存的函数。如果你需要可变的内存块,ctypes有一个create_string_buffer()以各种方式创建它们的 函数。可以使用raw 属性访问(或更改)当前内存块内容; 如果要以NUL终止字符串的形式访问它,请使用以下value 属性:

  1. >>> from ctypes import *
  2. >>> p = create_string_buffer(3) # create a 3 byte buffer, initialized to NUL bytes
  3. >>> print sizeof(p), repr(p.raw)
  4. 3 '\x00\x00\x00'
  5. >>> p = create_string_buffer("Hello") # create a buffer containing a NUL terminated string
  6. >>> print sizeof(p), repr(p.raw)
  7. 6 'Hello\x00'
  8. >>> print repr(p.value)
  9. 'Hello'
  10. >>> p = create_string_buffer("Hello", 10) # create a 10 byte buffer
  11. >>> print sizeof(p), repr(p.raw)
  12. 10 'Hello\x00\x00\x00\x00\x00'
  13. >>> p.value = "Hi"
  14. >>> print sizeof(p), repr(p.raw)
  15. 10 'Hi\x00lo\x00\x00\x00\x00\x00'
  16. >>>

create_string_buffer()函数替换了c_buffer()函数(仍可作为别名使用),以及c_string()早期ctypes版本中的函数。要创建包含C类型的unicode字符的可变内存块,请wchar_t使用该 create_unicode_buffer()函数。

   15.17.1.5。调用函数,续

需要注意的是的printf打印到实际的标准输出通道,给 sys.stdout,所以这些例子只是在控制台提示符下运行,而不是从内IDLEPythonWin的

  1. >>> printf = libc.printf
  2. >>> printf("Hello, %s\n", "World!")
  3. Hello, World!
  4. 14
  5. >>> printf("Hello, %S\n", u"World!")
  6. Hello, World!
  7. 14
  8. >>> printf("%d bottles of beer\n", 42)
  9. 42 bottles of beer
  10. 19
  11. >>> printf("%f bottles of beer\n", 42.5)
  12. Traceback (most recent call last):
  13. File "<stdin>", line 1, in <module>
  14. ArgumentError: argument 2: exceptions.TypeError: Don't know how to convert parameter 2
  15. >>>

如前所述,除了整数,字符串和unicode字符串之外的所有Python类型都必须包装在相应的ctypes类型中,以便它们可以转换为所需的C数据类型:

  1. >>> printf("An int %d, a double %f\n", 1234, c_double(3.14))
  2. An int 1234, a double 3.140000
  3. 31
  4. >>>

    15.17.1.6。使用自己的自定义数据类型调用函数

您还可以自定义ctypes参数转换,以允许将您自己的类的实例用作函数参数。 ctypes查找 _as_parameter_属性并将其用作函数参数。当然,它必须是整数,字符串或unicode之一:

  1. >>> class Bottles(object):
  2. ... def __init__(self, number):
  3. ... self._as_parameter_ = number
  4. ...
  5. >>> bottles = Bottles(42)
  6. >>> printf("%d bottles of beer\n", bottles)
  7. 42 bottles of beer
  8. 19
  9. >>>

如果您不想将实例的数据存储在_as_parameter_ 实例变量中,则可以定义property()使数据可用的数据。

   15.17.1.7。指定必需的参数类型(函数原型)

可以通过设置argtypes属性来指定从DLL导出的函数所需的参数类型。

argtypes必须是一系列C数据类型(这里的printf函数可能不是一个很好的例子,因为它取决于格式字符串需要一个可变数字和不同类型的参数,另一方面,这对于试验这个特性非常方便) :

  1. >>> printf.argtypes = [c_char_p, c_char_p, c_int, c_double]
  2. >>> printf("String '%s', Int %d, Double %f\n", "Hi", 10, 2.2)
  3. String 'Hi', Int 10, Double 2.200000
  4. 37
  5. >>>

指定格式可以防止不兼容的参数类型(就像C函数的原型一样),并尝试将参数转换为有效类型:

  1. >>> printf("%d %d %d", 1, 2, 3)
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. ArgumentError: argument 2: exceptions.TypeError: wrong type
  5. >>> printf("%s %d %f\n", "X", 2, 3)
  6. X 2 3.000000
  7. 13
  8. >>>

如果已经定义了自己传递给函数调用的from_param()类,则必须为它们实现一个类方法,以便能够在argtypes序列中使用它们。本from_param()类方法接收传给函数的Python对象,它做一个类型检测,或者是需要确保这个对象是可接受的,然后返回对象本身,它_as_parameter_不管你想传递的C函数属性,或在这种情况下的论点。同样,结果应该是整数,字符串,unicode,ctypes实例或具有_as_parameter_属性的对象 。

   15.17.1.8。返回类型

默认情况下,假定函数返回C int类型。可以通过设置restype函数对象的属性来指定其他返回类型。

这是一个更高级的示例,它使用strchr函数,它需要一个字符串指针和一个char,并返回一个指向字符串的指针:

  1. >>> strchr = libc.strchr
  2. >>> strchr("abcdef", ord("d"))
  3. 8059983
  4. >>> strchr.restype = c_char_p # c_char_p is a pointer to a string
  5. >>> strchr("abcdef", ord("d"))
  6. 'def'
  7. >>> print strchr("abcdef", ord("x"))
  8. None
  9. >>>

如果要避免ord("x")上面的调用,可以设置 argtypes属性,第二个参数将从单个字符Python字符串转换为C字符:

  1. >>> strchr.restype = c_char_p
  2. >>> strchr.argtypes = [c_char_p, c_char]
  3. >>> strchr("abcdef", "d")
  4. 'def'
  5. >>> strchr("abcdef", "def")
  6. Traceback (most recent call last):
  7. File "<stdin>", line 1, in <module>
  8. ArgumentError: argument 2: exceptions.TypeError: one character string expected
  9. >>> print strchr("abcdef", "x")
  10. None
  11. >>> strchr("abcdef", "d")
  12. 'def'
  13. >>>

restype如果外部函数返回一个整数,您还可以使用可调用的Python对象(例如函数或类)作为属性。将使用C函数返回的整数调用callable ,并且此调用的结果将用作函数调用的结果。这对于检查错误返回值并自动引发异常非常有用:

  1. >>> GetModuleHandle = windll.kernel32.GetModuleHandleA
  2. >>> def ValidHandle(value):
  3. ... if value == 0:
  4. ... raise WinError()
  5. ... return value
  6. ...
  7. >>>
  8. >>> GetModuleHandle.restype = ValidHandle
  9. >>> GetModuleHandle(None)
  10. 486539264
  11. >>> GetModuleHandle("something silly")
  12. Traceback (most recent call last):
  13. File "<stdin>", line 1, in <module>
  14. File "<stdin>", line 3, in ValidHandle
  15. WindowsError: [Errno 126] The specified module could not be found.
  16. >>>

WinError是一个函数,它将调用Windows FormatMessage()api来获取错误代码的字符串表示,并返回一个异常。 WinError采用可选的错误代码参数,如果没有使用,则调用它 GetLastError()来检索它。

请注意,通过该errcheck属性可以使用更强大的错误检查机制; 有关详细信息,请参阅参考手册

   15.17.1.9。传递指针(或:通过引用传递参数)

有时,C api函数需要将指向数据类型的指针作为参数,可能要写入相应的位置,或者数据太大而无法通过值传递。这也称为通过引用传递参数

ctypes导出byref()用于通过引用传递参数的函数。使用该pointer()函数可以实现相同的效果 ,但是pointer()由于它构造了一个真正的指针对象,所以工作量更大,因此byref()如果您不需要Python本身的指针对象,则使用它会更快:

  1. >>> i = c_int()
  2. >>> f = c_float()
  3. >>> s = create_string_buffer('\000' * 32)
  4. >>> print i.value, f.value, repr(s.value)
  5. 0 0.0 ''
  6. >>> libc.sscanf("1 3.14 Hello", "%d %f %s",
  7. ... byref(i), byref(f), s)
  8. 3
  9. >>> print i.value, f.value, repr(s.value)
  10. 1 3.1400001049 'Hello'
  11. >>>

   15.17.1.10。结构和联合

结构和联合必须从导出StructureUnion 其中所定义的基类ctypes模块。每个子类都必须定义一个_fields_属性。 _fields_必须是2元组的列表 ,包含字段名称字段类型

字段类型必须是ctypes类型c_int或任何其他派生ctypes类型:结构,联合,数组,指针。

下面是一个POINT结构的简单示例,它包含两个名为xy的整数 ,还显示了如何在构造函数中初始化结构:

  1. >>> from ctypes import *
  2. >>> class POINT(Structure):
  3. ... _fields_ = [("x", c_int),
  4. ... ("y", c_int)]
  5. ...
  6. >>> point = POINT(10, 20)
  7. >>> print point.x, point.y
  8. 10 20
  9. >>> point = POINT(y=5)
  10. >>> print point.x, point.y
  11. 0 5
  12. >>> POINT(1, 2, 3)
  13. Traceback (most recent call last):
  14. File "<stdin>", line 1, in <module>
  15. ValueError: too many initializers
  16. >>>

但是,您可以构建更复杂的结构。通过将结构用作字段类型,结构本身可以包含其他结构。

这是一个RECT结构,它包含两个名为upperleft和 lowerright的POINT

  1. >>> class RECT(Structure):
  2. ... _fields_ = [("upperleft", POINT),
  3. ... ("lowerright", POINT)]
  4. ...
  5. >>> rc = RECT(point)
  6. >>> print rc.upperleft.x, rc.upperleft.y
  7. 0 5
  8. >>> print rc.lowerright.x, rc.lowerright.y
  9. 0 0
  10. >>>

嵌套结构也可以通过以下几种方式在构造函数中初始化:

  1. >>> r = RECT(POINT(1, 2), POINT(3, 4))
  2. >>> r = RECT((1, 2), (3, 4))

字段描述符 S可从被检索,它们可用于调试有用,因为它们可以提供有用的信息:

  1. >>> print POINT.x
  2. <Field type=c_long, ofs=0, size=4>
  3. >>> print POINT.y
  4. <Field type=c_long, ofs=4, size=4>
  5. >>>

警告

ctypes不支持通过值将具有位字段的联合或结构传递给函数。虽然这可能适用于32位x86,但是不能保证库在一般情况下工作。具有位字段的联合和结构应始终通过指针传递给函数。

   15.17.1.11。结构/联合对齐和字节顺序

默认情况下,Structure和Union字段的对齐方式与C编译器的方式相同。可以通过_pack_在子类定义中指定类属性来覆盖此行为 。必须将其设置为正整数,并指定字段的最大对齐方式。这也是MSVC中的做法。#pragma pack(n)

ctypes使用结构和联合的本机字节顺序。要建立与非本地字节顺序结构,你可以使用一个 BigEndianStructureLittleEndianStructure, BigEndianUnion,和LittleEndianUnion基类。这些类不能包含指针字段。

   15.17.1.12。结构和联合中的位字段

可以创建包含位字段的结构和联合。位字段仅适用于整数字段,位宽指定为_fields_元组中的第三项:

  1. >>> class Int(Structure):
  2. ... _fields_ = [("first_16", c_int, 16),
  3. ... ("second_16", c_int, 16)]
  4. ...
  5. >>> print Int.first_16
  6. <Field type=c_long, ofs=0:0, bits=16>
  7. >>> print Int.second_16
  8. <Field type=c_long, ofs=0:16, bits=16>
  9. >>>

   15.17.1.13。数组

数组是序列,包含固定数量的相同类型的实例。

创建数组类型的推荐方法是将数据类型与正整数相乘:

TenPointsArrayType = POINT * 10

这是一个有点人为的数据类型的例子,一个包含4个POINT和其他东西的结构:

  1. >>> from ctypes import *
  2. >>> class POINT(Structure):
  3. ... _fields_ = ("x", c_int), ("y", c_int)
  4. ...
  5. >>> class MyStruct(Structure):
  6. ... _fields_ = [("a", c_int),
  7. ... ("b", c_float),
  8. ... ("point_array", POINT * 4)]
  9. >>>
  10. >>> print len(MyStruct().point_array)
  11. 4
  12. >>>

通过调用类以通常的方式创建实例:

  1. arr = TenPointsArrayType()
  2. for pt in arr:
  3. print pt.x, pt.y

上面的代码打印了一系列行,因为数组内容被初始化为零。0 0

也可以指定正确类型的初始化器:

  1. >>> from ctypes import *
  2. >>> TenIntegers = c_int * 10
  3. >>> ii = TenIntegers(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
  4. >>> print ii
  5. <c_long_Array_10 object at 0x...>
  6. >>> for i in ii: print i,
  7. ...
  8. 1 2 3 4 5 6 7 8 9 10
  9. >>>

   15.17.1.14。指针

通过pointer()ctypes类型上调用函数 来创建指针实例:

  1. >>> from ctypes import *
  2. >>> i = c_int(42)
  3. >>> pi = pointer(i)
  4. >>>

指针实例有一个contents属性,它返回指针指向的i对象,上面的对象:

  1. >>> pi.contents
  2. c_long(42)
  3. >>>

注意,ctypes没有OOR(原始对象返回),每次检索属性时它都会构造一个新的等效对象:

  1. >>> pi.contents is i
  2. False
  3. >>> pi.contents is pi.contents
  4. False
  5. >>>

将另一个c_int实例分配给指针的contents属性会导致指针指向存储它的内存位置:

  1. >>> i = c_int(99)
  2. >>> pi.contents = i
  3. >>> pi.contents
  4. c_long(99)
  5. >>>

指针实例也可以用整数索引:

  1. >>> pi[0]
  2. 99
  3. >>>

分配整数索引会更改指向的值:

  1. >>> print i
  2. c_long(99)
  3. >>> pi[0] = 22
  4. >>> print i
  5. c_long(22)
  6. >>>

也可以使用不同于0的索引,但您必须知道自己在做什么,就像在C中一样:您可以访问或更改任意内存位置。通常,如果从C函数接收指针,则只使用此功能,并且您知道指针实际指向的是数组而不是单个项。

在幕后,该pointer()函数不仅仅是创建指针实例,还必须首先创建指针类型。这是通过POINTER()接受任何ctypes类型的函数完成的,并返回一个新类型:

  1. >>> PI = POINTER(c_int)
  2. >>> PI
  3. <class 'ctypes.LP_c_long'>
  4. >>> PI(42)
  5. Traceback (most recent call last):
  6. File "<stdin>", line 1, in <module>
  7. TypeError: expected c_long instead of int
  8. >>> PI(c_int(42))
  9. <ctypes.LP_c_long object at 0x...>
  10. >>>

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

  1. >>> null_ptr = POINTER(c_int)()
  2. >>> print bool(null_ptr)
  3. False
  4. >>>

ctypes检查NULL何时解除引用指针(但解除引用无效的非NULL指针会使Python崩溃):

  1. >>> null_ptr[0]
  2. Traceback (most recent call last):
  3. ....
  4. ValueError: NULL pointer access
  5. >>>
  6. >>> null_ptr[0] = 1234
  7. Traceback (most recent call last):
  8. ....
  9. ValueError: NULL pointer access
  10. >>>

   15.17.1.15。输入转换

通常,ctypes会进行严格的类型检查。这意味着,如果您 POINTER(c_int)argtypes函数列表中或在结构定义中具有成员字段的类型,则只接受完全相同类型的实例。此规则有一些例外,其中ctypes接受其他对象。例如,您可以传递兼容的数组实例而不是指针类型。因此,对于POINTER(c_int),ctypes接受一个c_int数组:

  1. >>> class Bar(Structure):
  2. ... _fields_ = [("count", c_int), ("values", POINTER(c_int))]
  3. ...
  4. >>> bar = Bar()
  5. >>> bar.values = (c_int * 3)(1, 2, 3)
  6. >>> bar.count = 3
  7. >>> for i in range(bar.count):
  8. ... print bar.values[i]
  9. ...
  10. 1
  11. 2
  12. 3
  13. >>>

此外,如果函数参数显式声明为指针类型(例如POINTER(c_int))in argtypes,则可以将指向类型的对象(c_int在本例中)传递给函数。ctypes将自动应用所需的byref()转换。

要将POINTER类型字段设置为NULL,您可以指定None

  1. >>> bar.values = None
  2. >>>

有时您会遇到不兼容类型的实例。在C中,您可以将一种类型转换为另一种类型。 ctypes提供cast()可以以相同方式使用的功能。Bar上面定义的结构接受 其字段的POINTER(c_int)指针或c_int数组values,但不接受其他类型的实例:

  1. >>> bar.values = (c_byte * 4)()
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. TypeError: incompatible types, c_byte_Array_4 instance instead of LP_c_long instance
  5. >>>

对于这些情况,该cast()功能很方便,cast还支持将python中的id(obj)即对象地址还原为对象

cast()函数可用于将ctypes实例转换为指向不同ctypes数据类型的指针。 cast()采用两个参数,一个或者可以转换为某种指针的ctypes对象,以及一个ctypes指针类型。它返回第二个参数的实例,它引用与第一个参数相同的内存块:

  1. >>> a = (c_byte * 4)()
  2. >>> cast(a, POINTER(c_int))
  3. <ctypes.LP_c_long object at ...>
  4. >>>

所以,cast()可以用来分配给结构的values字段Bar

  1. >>> bar = Bar()
  2. >>> bar.values = cast((c_byte * 4)(), POINTER(c_int))
  3. >>> print bar.values[0]
  4. 0
  5. >>>

   15.17.1.16。不完全类型

不完整类型是其成员尚未指定的结构,联合或数组。在C中,它们由前向声明指定,后面将定义:

  1. struct cell; /* forward declaration */
  2. struct cell {
  3. char *name;
  4. struct cell *next;
  5. };

直接转换为ctypes代码就是这样,但它不起作用:

  1. >>> class cell(Structure):
  2. ... _fields_ = [("name", c_char_p),
  3. ... ("next", POINTER(cell))]
  4. ...
  5. Traceback (most recent call last):
  6. File "<stdin>", line 1, in <module>
  7. File "<stdin>", line 2, in cell
  8. NameError: name 'cell' is not defined
  9. >>>

因为new 语句本身不可用。在,我们可以 在类语句之后定义类并设置属性:class cellctypescell_fields_

  1. >>> from ctypes import *
  2. >>> class cell(Structure):
  3. ... pass
  4. ...
  5. >>> cell._fields_ = [("name", c_char_p),
  6. ... ("next", POINTER(cell))]
  7. >>>

让我们试试吧。我们创建了两个实例cell,并让它们相互指向,最后跟随指针链几次:

  1. >>> c1 = cell()
  2. >>> c1.name = "foo"
  3. >>> c2 = cell()
  4. >>> c2.name = "bar"
  5. >>> c1.next = pointer(c2)
  6. >>> c2.next = pointer(c1)
  7. >>> p = c1
  8. >>> for i in range(8):
  9. ... print p.name,
  10. ... p = p.next[0]
  11. ...
  12. foo bar foo bar foo bar foo bar
  13. >>>

   15.17.1.17。回调函数

ctypes允许从Python callables创建C可调用函数指针。这些有时称为回调函数

首先,您必须为回调函数创建一个类,该类知道调用约定,返回类型以及此函数将接收的参数的数量和类型。

CFUNCTYPE工厂函数使用普通的cdecl调用约定为回调函数创建类型,在Windows上,WINFUNCTYPE工厂函数使用stdcall调用约定为回调函数创建类型。

这两个工厂函数都以结果类型作为第一个参数调用,而回调函数将期望的参数类型作为其余参数。

我将在这里展示一个使用标准C库qsort() 函数的示例,它用于在回调函数的帮助下对项目进行排序。 qsort()将用于对整数数组进行排序:

  1. >>> IntArray5 = c_int * 5
  2. >>> ia = IntArray5(5, 1, 7, 33, 99)
  3. >>> qsort = libc.qsort
  4. >>> qsort.restype = None
  5. >>>

qsort()必须使用指向要排序的数据的指针,数据数组中的项数,一个项的大小以及指向比较函数(回调)的指针来调用。然后使用两个指向项目的指针调用回调,如果第一个项目小于第二个项目,它必须返回一个负整数,如果它们相等则返回零,然后返回一个正整数。

所以我们的回调函数接收指向整数的指针,并且必须返回一个整数。首先我们type为回调函数创建:

  1. >>> CMPFUNC = CFUNCTYPE(c_int, POINTER(c_int), POINTER(c_int))
  2. >>>

对于回调函数的第一个实现,我们只是打印我们得到的参数,并返回0(增量开发;-):

  1. >>> def py_cmp_func(a, b):
  2. ... print "py_cmp_func", a, b
  3. ... return 0
  4. ...
  5. >>>

创建C可调用回调:

  1. >>> cmp_func = CMPFUNC(py_cmp_func)
  2. >>>

我们准备好了:

  1. >>> qsort(ia, len(ia), sizeof(c_int), cmp_func)
  2. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  3. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  4. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  5. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  6. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  7. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  8. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  9. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  10. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  11. py_cmp_func <ctypes.LP_c_long object at 0x00...> <ctypes.LP_c_long object at 0x00...>
  12. >>>

我们知道如何访问指针的内容,所以让我们重新定义我们的回调:

  1. >>> def py_cmp_func(a, b):
  2. ... print "py_cmp_func", a[0], b[0]
  3. ... return 0
  4. ...
  5. >>> cmp_func = CMPFUNC(py_cmp_func)
  6. >>>

以下是我们在Windows上获得的内容:

  1. >>> qsort(ia, len(ia), sizeof(c_int), cmp_func)
  2. py_cmp_func 7 1
  3. py_cmp_func 33 1
  4. py_cmp_func 99 1
  5. py_cmp_func 5 1
  6. py_cmp_func 7 5
  7. py_cmp_func 33 5
  8. py_cmp_func 99 5
  9. py_cmp_func 7 99
  10. py_cmp_func 33 99
  11. py_cmp_func 7 33
  12. >>>

很有趣的是,在linux上,sort函数看起来效率更高,它做的比较少:

  1. >>> qsort(ia, len(ia), sizeof(c_int), cmp_func)
  2. py_cmp_func 5 1
  3. py_cmp_func 33 99
  4. py_cmp_func 7 33
  5. py_cmp_func 5 7
  6. py_cmp_func 1 7
  7. >>>

啊,我们差不多完成了!最后一步是实际比较这两个项并返回一个有用的结果:

  1. >>> def py_cmp_func(a, b):
  2. ... print "py_cmp_func", a[0], b[0]
  3. ... return a[0] - b[0]
  4. ...
  5. >>>

最终在Windows上运行:

  1. >>> qsort(ia, len(ia), sizeof(c_int), CMPFUNC(py_cmp_func))
  2. py_cmp_func 33 7
  3. py_cmp_func 99 33
  4. py_cmp_func 5 99
  5. py_cmp_func 1 99
  6. py_cmp_func 33 7
  7. py_cmp_func 1 33
  8. py_cmp_func 5 33
  9. py_cmp_func 5 7
  10. py_cmp_func 1 7
  11. py_cmp_func 5 1
  12. >>>

在Linux上:

  1. >>> qsort(ia, len(ia), sizeof(c_int), CMPFUNC(py_cmp_func))
  2. py_cmp_func 5 1
  3. py_cmp_func 33 99
  4. py_cmp_func 7 33
  5. py_cmp_func 1 7
  6. py_cmp_func 5 7
  7. >>>

很有趣的是,Windows qsort()功能需要比Linux版本更多的比较!

我们可以轻松检查,我们的数组现在排序:

  1. >>> for i in ia: print i,
  2. ...
  3. 1 5 7 33 99
  4. >>>

注意

CFUNCTYPE()只要从C代码中使用对象,请确保保留对对象的引用。ctypes没有,如果你不这样做,它们可能被垃圾收集,在回调时崩溃你的程序。

另外,请注意,如果在Python控件之外创建的线程中调用回调函数(例如,通过调用回调的外部代码),ctypes会在每次调用时创建一个新的虚拟Python线程。这种行为是对大多数的目的是正确的,但它意味着存储的值threading.local不会在不同的回调生存,即使这些电话是从同一个C的线。

   15.17.1.18。访问从dll导出的值

一些共享库不仅导出函数,还导出变量。Python库本身的一个示例是Py_OptimizeFlag,一个设置为0,1或2的整数,具体取决于启动时给出的-O-OO标志。

ctypes可以使用in_dll()类型的类方法访问这样的值。 pythonapi是一个预定义的符号,可以访问Python C api:

  1. >>> opt_flag = c_int.in_dll(pythonapi, "Py_OptimizeFlag")
  2. >>> print opt_flag
  3. c_long(0)
  4. >>>

如果解释器已经启动-O,则样本将打印c_long(1),或者c_long(2)如果-OO已经指定。

一个扩展示例也演示了指针的使用,可以访问PyImport_FrozenModulesPython导出的 指针。

引用Python文档:此指针初始化为指向“struct _frozen”记录的数组,由其成员全部为NULL或零的记录终止。导入冻结模块时,将在此表中搜索它。第三方代码可以使用此方法来提供动态创建的冻结模块集合。

所以操纵这个指针甚至可以证明是有用的。为了限制示例大小,我们仅显示如何使用以下方法读取此表ctypes

  1. >>> from ctypes import *
  2. >>>
  3. >>> class struct_frozen(Structure):
  4. ... _fields_ = [("name", c_char_p),
  5. ... ("code", POINTER(c_ubyte)),
  6. ... ("size", c_int)]
  7. ...
  8. >>>

我们已经定义了数据类型,因此我们可以获得指向表的指针:struct _frozen

  1. >>> FrozenTable = POINTER(struct_frozen)
  2. >>> table = FrozenTable.in_dll(pythonapi, "PyImport_FrozenModules")
  3. >>>

因为table是一个记录pointer数组struct_frozen,我们可以迭代它,但我们必须确保我们的循环终止,因为指针没有大小。迟早它可能会因访问冲突或其他原因而崩溃,所以当我们点击NULL条目时最好突破循环:

  1. >>> for item in table:
  2. ... print item.name, item.size
  3. ... if item.name is None:
  4. ... break
  5. ...
  6. __hello__ 104
  7. __phello__ -104
  8. __phello__.spam 104
  9. None 0
  10. >>>

标准Python具有冻结模块和冻结包(由负大小成员指示)的事实并不为人所知,它仅用于测试。例如尝试一下。import __hello__

   15.17.1.19。惊喜

在某些边缘情况下ctypes,您可能会发现实际情况以外的其他情况。

请考虑以下示例:

  1. >>> from ctypes import *
  2. >>> class POINT(Structure):
  3. ... _fields_ = ("x", c_int), ("y", c_int)
  4. ...
  5. >>> class RECT(Structure):
  6. ... _fields_ = ("a", POINT), ("b", POINT)
  7. ...
  8. >>> p1 = POINT(1, 2)
  9. >>> p2 = POINT(3, 4)
  10. >>> rc = RECT(p1, p2)
  11. >>> print rc.a.x, rc.a.y, rc.b.x, rc.b.y
  12. 1 2 3 4
  13. >>> # now swap the two points
  14. >>> rc.a, rc.b = rc.b, rc.a
  15. >>> print rc.a.x, rc.a.y, rc.b.x, rc.b.y
  16. 3 4 3 4
  17. >>>

嗯。我们当然希望打印最后一份声明。发生了什么?以下是上述行的步骤:3 4 1 2rc.a, rc.b = rc.b, rc.a

  1. >>> temp0, temp1 = rc.b, rc.a
  2. >>> rc.a = temp0
  3. >>> rc.b = temp1
  4. >>>

请注意,temp0并且temp1还在使用的内部缓冲器的对象rc上述对象。因此执行将缓冲区内容复制到缓冲区中。反过来,这改变了内容。因此,最后一项任务,没有预期的效果。rc.a = temp0temp0rctemp1rc.b = temp1

请记住,从Structure,Unions和Arrays中检索子对象不会复制子对象,而是检索访问根对象底层缓冲区的包装器对象。

另一个可能与预期不同的例子是:

  1. >>> s = c_char_p()
  2. >>> s.value = "abc def ghi"
  3. >>> s.value
  4. 'abc def ghi'
  5. >>> s.value is s.value
  6. False
  7. >>>

为什么打印False?ctypes实例是包含内存块的对象以及访问内存内容的一些描述符。在内存块中存储Python对象不会存储对象本身,而是存储对象contents的对象。再次访问内容每次构造一个新的Python对象!

   15.17.1.20。可变大小的数据类型

ctypes 为可变大小的数组和结构提供了一些支持。

resize()函数可用于调整现有ctypes对象的内存缓冲区的大小。该函数将对象作为第一个参数,并将请求的大小(以字节为单位)作为第二个参数。内存块不能小于对象类型指定的自然内存块,ValueError如果尝试,则引发a :

  1. >>> short_array = (c_short * 4)()
  2. >>> print sizeof(short_array)
  3. 8
  4. >>> resize(short_array, 4)
  5. Traceback (most recent call last):
  6. ...
  7. ValueError: minimum size is 8
  8. >>> resize(short_array, 32)
  9. >>> sizeof(short_array)
  10. 32
  11. >>> sizeof(type(short_array))
  12. 8
  13. >>>

这很好很好,但是如何访问此数组中包含的其他元素?由于类型仍然只知道4个元素,因此访问其他元素时会出错:

  1. >>> short_array[:]
  2. [0, 0, 0, 0]
  3. >>> short_array[7]
  4. Traceback (most recent call last):
  5. ...
  6. IndexError: invalid index
  7. >>>

使用可变大小数据类型的另一种方法ctypes是使用Python的动态特性,并且在已知所需大小之后(根据具体情况)重新定义数据类型。

  15.17.2。ctypes引用

  15.17.2.1。查找共享库

使用编译语言编程时,在编译/链接程序时以及程序运行时都会访问共享库。

find_library()函数的目的是以类似于编译器的方式定位库(在具有多个版本的共享库的平台上应该加载最新版本),而ctypes库加载器就像程序运行时一样,并直接调用运行时加载程序。

ctypes.util模块提供了一个函数,可以帮助确定要加载的库。

ctypes.util.find_library姓名

尝试查找库并返回路径名。 名字是不一样的任何前缀库名的lib,标的相同.so.dylib或版本号(这是用于POSIX链接器选项的形式-l)。如果找不到库,则返回None

确切的功能取决于系统。

在Linux上,find_library()尝试运行外部程序(/sbin/ldconfiggcc,和objdump)找到库文件。它返回库文件的文件名。这里有些例子:

  1. >>> from ctypes.util import find_library
  2. >>> find_library("m")
  3. 'libm.so.6'
  4. >>> find_library("c")
  5. 'libc.so.6'
  6. >>> find_library("bz2")
  7. 'libbz2.so.1.0'
  8. >>>

在OS X上,find_library()尝试几个预定义的命名方案和路径来定位库,如果成功则返回完整路径名:

  1. >>> from ctypes.util import find_library
  2. >>> find_library("c")
  3. '/usr/lib/libc.dylib'
  4. >>> find_library("m")
  5. '/usr/lib/libm.dylib'
  6. >>> find_library("bz2")
  7. '/usr/lib/libbz2.dylib'
  8. >>> find_library("AGL")
  9. '/System/Library/Frameworks/AGL.framework/AGL'
  10. >>>

在Windows上,find_library()沿系统搜索路径进行搜索,并返回完整路径名,但由于没有预定义的命名方案,因此调用find_library("c")将失败并返回None

如果包装共享库ctypes,它可以更好地确定在开发时共享库的名字,并硬编码到封装模块,而不是使用find_library()定位在运行时库。

   15.17.2.2。加载共享库

有几种方法可以将共享库加载到Python进程中。一种方法是实例化以下类之一:

class ctypes.CDLLnamemode = DEFAULT_MODEhandle = Noneuse_errno = Falseuse_last_error = False 

此类的实例表示已加载的共享库。这些库中的函数使用标准C调用约定,并假定返回 int

class ctypes.OleDLLnamemode = DEFAULT_MODEhandle = Noneuse_errno = Falseuse_last_error = False 

仅限Windows:此类的实例表示已加载的共享库,这些库中的函数使用stdcall调用约定,并假定返回特定于Windows的HRESULT代码。 HRESULT values包含指定函数调用是否失败或成功的信息,以及其他错误代码。如果返回值表示失败,WindowsError则自动引发a。

class ctypes.WinDLLnamemode = DEFAULT_MODEhandle = Noneuse_errno = Falseuse_last_error = False 

仅限Windows:此类的实例表示已加载的共享库,这些库中的函数使用stdcall调用约定,并且int默认情况下假定返回。

在Windows CE上,仅使用标准调用约定,以方便 WinDLLOleDLL使用此平台上的标准调用约定。

Python 全局解释器锁在调用这些库导出的任何函数之前发布,之后重新获取。

class ctypes.PyDLLnamemode = DEFAULT_MODEhandle = None 

CDLL除了在函数调用期间释放Python GIL之外,此类的实例的行为与实例类似,并且在函数执行之后,将检查Python错误标志。如果设置了错误标志,则会引发Python异常。

因此,这仅对直接调用Python C api函数有用。

所有这些类都可以通过使用至少一个参数(共享库的路径名)调用它们来实例化。如果已有已加载的共享库的现有句柄,则可以将其作为handle命名参数传递,否则使用基础平台dlopenLoadLibrary 函数将库加载到进程中,并获取它的句柄。

模式参数可用于指定库的加载方式。有关详细信息,请参阅dlopen(3)联机帮助页。在Windows上,模式被忽略。在posix系统上,始终添加RTLD_NOW,并且不可配置。

use_errno参数,当设置为true,使一个ctypes机制,允许访问该系统errno以安全的方式错误号。 ctypes维护系统errno 变量的线程局部副本; 如果在函数调用与ctypes私有副本交换之前调用用函数调用创建的外部函数,use_errno=True则在 errno函数调用之后立即发生相同的操作。

该函数ctypes.get_errno()返回ctypes私有副本的值,该函数将ctypes私有副本ctypes.set_errno()更改为新值并返回前一个值。

use_last_error参数,设置为true时,使能由所述管理Windows错误代码相同的机制GetLastError()和 SetLastError()Windows API函数; ctypes.get_last_error()并 ctypes.set_last_error()用于请求和更改Windows错误代码的ctypes私有副本。

新的2.6版:use_last_erroruse_errno可选参数添加。

ctypes.RTLD_GLOBAL

用作模式参数的标志。在此标志不可用的平台上,它被定义为整数零。

ctypes.RTLD_LOCAL

用作模式参数的标志。在没有此功能的平台上,它与RTLD_GLOBAL相同。

ctypes.DEFAULT_MODE

用于加载共享库的默认模式。在OSX 10.3上,这是 RTLD_GLOBAL,否则它与RTLD_LOCAL相同。

这些类的实例没有公共方法。共享库导出的函数可以作为属性或索引进行访问。请注意,通过属性访问函数会缓存结果,因此每次重复访问它都会返回相同的对象。另一方面,通过索引访问它每次都会返回一个新对象:

  1. >>> libc.time == libc.time
  2. True
  3. >>> libc['time'] == libc['time']
  4. False

可以使用以下公共属性,它们的名称以下划线开头,以便不与导出的函数名冲突:

PyDLL._handle

用于访问库的系统句柄。

PyDLL._name

在构造函数中传递的库的名称。

还可以使用其中一个预制对象(LibraryLoader通过调用 LoadLibrary()方法)或通过将库检索为加载程序实例的属性来加载共享库。

class ctypes.LibraryLoaderdlltype 

加载共享库的类。 dlltype应该是一个 CDLLPyDLLWinDLLOleDLL类型。

__getattr__()具有特殊行为:它允许通过将其作为库加载器实例的属性访问来加载共享库。结果是缓存的,因此重复的属性访问每次都返回相同的库。

LoadLibrary名字

将共享库加载到进程中并返回它。此方法始终返回库的新实例。

这些预制库加载器可用:

ctypes.cdll

创建CDLL实例。

ctypes.windll

仅限Windows:创建WinDLL实例。

ctypes.oledll

仅限Windows:创建OleDLL实例。

ctypes.pydll

创建PyDLL实例。

为了直接访问C Python api,可以使用现成的Python共享库对象:

ctypes.pythonapi

它的一个实例PyDLL将Python C API函数公开为属性。请注意,假设所有这些函数都返回C int,这当然不总是事实,因此您必须分配正确的restype属性才能使用这些函数。

   15.17.2.3。外来函数

如前一节所述,外部函数可以作为加载的共享库的属性进行访问。默认情况下以这种方式创建的函数对象接受任意数量的参数,接受任何ctypes数据实例作为参数,并返回库加载器指定的默认结果类型。他们是私人班级的实例:

ctypes._FuncPtr

C可调用外部函数的基类。

外部函数的实例也是C兼容的数据类型; 它们代表C函数指针。

可以通过分配外部函数对象的特殊属性来自定义此行为。

restype

指定ctypes类型以指定外部函数的结果类型。使用Nonevoid,功能不返回任何东西。

可以分配不是ctypes类型的可调用Python对象,在这种情况下假定函数返回C int,并且将使用此整数调用callable,从而允许进一步处理或错误检查。不推荐使用它,为了更灵活的后处理或错误检查,请使用ctypes数据类型, restype并为该errcheck属性分配一个callable 。

argtypes

分配ctypes类型的元组以指定函数接受的参数类型。使用stdcall调用约定的函数只能使用与此元组的长度相同的参数数来调用; 使用C调用约定的函数也接受其他未指定的参数。

当调用外部函数时,每个实际参数都传递给 元组中from_param()项的 类方法argtypes,此方法允许将实际参数调整为外部函数接受的对象。例如,元组中的c_char_pargtypes将使用ctypes转换规则将作为参数传递的unicode字符串转换为字节字符串。

新增:现在可以将项目放在不是ctypes类型的argtypes中,但每个项目必须有一个from_param()返回可用作参数的值的方法(整数,字符串,ctypes实例)。这允许定义可以将自定义对象调整为函数参数的适配器。

errcheck

为此属性分配Python函数或其他可调用函数。将使用三个或更多参数调用callable:

callable结果函数参数

result是外部函数返回的内容,由restype属性指定 。

func是外部函数对象本身,这允许重用相同的可调用对象来检查或后处理几个函数的结果。

arguments是一个包含最初传递给函数调用的参数的元组,这允许对所使用的参数进行特殊处理。

此函数返回的对象将从外部函数调用返回,但如果外部函数调用失败,它还可以检查结果值并引发异常。

异常ctypes.ArgumentError

当外部函数调用无法转换其中一个传递的参数时,会引发此异常。

   15.17.2.4。函数原型

也可以通过实例化函数原型来创建外部函数。函数原型类似于C中的函数原型; 它们描述了一个函数(返回类型,参数类型,调用约定)而没有定义实现。必须使用所需的结果类型和函数的参数类型调用工厂函数。

ctypes.CFUNCTYPErestype* argtypesuse_errno = Falseuse_last_error = False 

返回的函数原型创建使用标准C调用约定的函数。该功能将在通话期间释放GIL。如果 use_errno设置为true,则系统errno变量的ctypes私有副本将 与errno调用前后的实际值进行交换; use_last_error对Windows错误代码执行相同操作。

版本2.6中已更改:添加了可选的use_errnouse_last_error参数。

ctypes.WINFUNCTYPErestype* argtypesuse_errno = Falseuse_last_error = False 

仅适用于Windows:返回的函数原型创建使用函数 stdcall调用约定,除了Windows CE地方 WINFUNCTYPE()是一样的CFUNCTYPE()。该功能将在通话期间释放GIL。 use_errnouse_last_error具有与上面相同的含义。

ctypes.PYFUNCTYPErestype* argtypes 

返回的函数原型创建使用Python调用约定的函数。该功能在通话期间不会释放GIL。

由这些工厂函数创建的函数原型可以以不同的方式实例化,具体取决于调用中参数的类型和数量:

prototype地址

返回指定地址的外部函数,该函数必须是整数。

prototype可赎回

从Python 可调用创建C可调用函数(回调函数)。

prototypefunc_spec [,paramflags ] )

返回由共享库导出的外部函数。func_spec必须是2元组。第一项是导出函数的名称作为字符串,或导出函数的序号作为小整数。第二项是共享库实例。(name_or_ordinal, library)

prototypevtbl_indexname [,paramflags [,iid ] ] )

返回将调用COM方法的外部函数。vtbl_index是虚函数表的索引,是一个小的非负整数。name是COM方法的名称。iid是指向扩展错误报告中使用的接口标识符的可选指针。

COM方法使用特殊的调用约定:除了argtypes元组中指定的那些参数外,它们还需要指向COM接口的指针作为第一个参数。

可选的paramflags参数创建的外部函数包装器具有比上述功能更多的功能。

paramflags必须是一个长度相同的元组argtypes

此元组中的每个项目都包含有关参数的更多信息,它必须是包含一个,两个或三个项目的元组。

第一项是一个整数,包含参数的方向标志组合:

1

指定函数的输入参数。

2

输出参数。外来函数填入一个值。

4

输入参数,默认为整数零。

可选的第二项是参数名称为字符串。如果指定了此参数,则可以使用命名参数调用外部函数。

可选的第三项是此参数的默认值。

此示例演示如何包装Windows MessageBoxA函数,以便它支持默认参数和命名参数。Windows头文件中的C声明是这样的:

  1. WINUSERAPI int WINAPI
  2. MessageBoxA(
  3. HWND hWnd,
  4. LPCSTR lpText,
  5. LPCSTR lpCaption,
  6. UINT uType);

这是包装ctypes

  1. >>> from ctypes import c_int, WINFUNCTYPE, windll
  2. >>> from ctypes.wintypes import HWND, LPCSTR, UINT
  3. >>> prototype = WINFUNCTYPE(c_int, HWND, LPCSTR, LPCSTR, UINT)
  4. >>> paramflags = (1, "hwnd", 0), (1, "text", "Hi"), (1, "caption", None), (1, "flags", 0)
  5. >>> MessageBox = prototype(("MessageBoxA", windll.user32), paramflags)
  6. >>>

现在可以通过以下方式调用MessageBox外部函数:

  1. >>> MessageBox()
  2. >>> MessageBox(text="Spam, spam, spam")
  3. >>> MessageBox(flags=2, text="foo bar")
  4. >>>

第二个例子演示了输出参数。win32 GetWindowRect 函数通过将指定窗口的尺寸复制到RECT调用者必须提供的结构中来检索它们的尺寸 。这是C声明:

  1. WINUSERAPI BOOL WINAPI
  2. GetWindowRect(
  3. HWND hWnd,
  4. LPRECT lpRect);

这是包装ctypes

  1. >>> from ctypes import POINTER, WINFUNCTYPE, windll, WinError
  2. >>> from ctypes.wintypes import BOOL, HWND, RECT
  3. >>> prototype = WINFUNCTYPE(BOOL, HWND, POINTER(RECT))
  4. >>> paramflags = (1, "hwnd"), (2, "lprect")
  5. >>> GetWindowRect = prototype(("GetWindowRect", windll.user32), paramflags)
  6. >>>

具有输出参数的函数将自动返回输出参数值(如果存在单个参数值)或者包含输出参数值的元组(如果有多个),则GetWindowRect函数现在会在调用时返回RECT实例。

输出参数可以与errcheck协议组合以进行进一步的输出处理和错误检查。win32 GetWindowRectapi函数返回一个BOOL指示成功或失败的信号,因此该函数可以执行错误检查,并在api调用失败时引发异常:

  1. >>> def errcheck(result, func, args):
  2. ... if not result:
  3. ... raise WinError()
  4. ... return args
  5. ...
  6. >>> GetWindowRect.errcheck = errcheck
  7. >>>

如果errcheck函数返回参数tuple,它接收不变,ctypes继续对输出参数进行的正常处理。如果要返回窗口坐标而不是RECT实例的元组 ,可以检索函数中的字段并返回它们,将不再进行正常处理:

  1. >>> def errcheck(result, func, args):
  2. ... if not result:
  3. ... raise WinError()
  4. ... rc = args[1]
  5. ... return rc.left, rc.top, rc.bottom, rc.right
  6. ...
  7. >>> GetWindowRect.errcheck = errcheck
  8. >>>

   15.17.2.5。实用功能

ctypes.addressofobj 

以整数形式返回内存缓冲区的地址。 obj必须是ctypes类型的实例。

ctypes.alignmentobj_or_type 

返回ctypes类型的对齐要求。obj_or_type必须是ctypes类型或实例。

ctypes.byrefobj [,offset ] )

返回一个指向obj的轻量级指针,该指针必须是ctypes类型的一个实例。 offset默认为零,并且必须是将添加到内部指针值的整数。

byref(obj, offset) 对应于此C代码:

(((char *)&obj) + offset)

返回的对象只能用作外部函数调用参数。它的行为类似pointer(obj),但构造速度要快得多。

:在2.6版本中的新偏移添加可选的参数。

ctypes.castobjtype 

此函数类似于C中的强制转换运算符。它返回一个新的类型实例,它指向与obj相同的内存块。 type 必须是指针类型,obj必须是可以解释为指针的对象。

ctypes.create_string_bufferinit_or_size [,size ] )

此函数创建一个可变字符缓冲区。返回的对象是ctypes数组c_char

init_or_size必须是一个整数,它指定数组的大小,或者是一个用于初始化数组项的字符串。

如果将字符串指定为第一个参数,则将缓冲区设置为大于字符串长度的一个项目,以便数组中的最后一个元素是NUL终止字符。可以将整数作为第二个参数传递,如果不应使用字符串的长度,则允许指定数组的大小。

如果第一个参数是unicode字符串,则根据ctypes转换规则将其转换为8位字符串。

ctypes.create_unicode_bufferinit_or_size [,size ] )

此函数创建一个可变的unicode字符缓冲区。返回的对象是ctypes数组c_wchar

init_or_size必须是指定数组大小的整数,或者是用于初始化数组项的unicode字符串。

如果将unicode字符串指定为第一个参数,则将缓冲区设置为大于字符串长度的一个项目,以便数组中的最后一个元素是NUL终止字符。可以将整数作为第二个参数传递,如果不应使用字符串的长度,则允许指定数组的大小。

如果第一个参数是8位字符串,则根据ctypes转换规则将其转换为unicode字符串。

ctypes.DllCanUnloadNow()

仅限Windows:此函数是一个钩子,允许使用ctypes实现进程内COM服务器。从DllCanUnloadNow函数调用_ctypes扩展dll导出。

ctypes.DllGetClassObject()

仅限Windows:此函数是一个钩子,允许使用ctypes实现进程内COM服务器。它是从_ctypes扩展dll导出的DllGetClassObject函数调用的。

ctypes.util.find_library名字

尝试查找库并返回路径名。 name是没有任何前缀的库名,如lib后缀.so.dylib或版本号(这是用于posix链接器选项的形式-l)。如果找不到库,则返回None

确切的功能取决于系统。

在版本2.6中更改:仅限Windows:find_library("m")find_library("c")将调用的结果返回到find_msvcrt()

ctypes.util.find_msvcrt()

仅限Windows:返回Python使用的VC运行时库的文件名,以及扩展模块。如果无法确定库的名称,None则返回。

如果需要释放内存,例如,由扩展模块调用的内存,则必须在分配内存的同一库中使用该功能。free(void *)

版本2.6中的新功能。

ctypes.FormatError([ code ] )

仅适用于Windows:返回的错误代码的文本描述代码。如果未指定错误代码,则通过调用Windows api函数GetLastError来使用上一个错误代码。

ctypes.GetLastError()

仅限Windows:返回Windows在调用线程中设置的最后一个错误代码。此函数直接调用Windows GetLastError()函数,它不返回错误代码的ctypes-private副本。

ctypes.get_errno()

返回errno调用线程中系统变量的ctypes-private副本的当前值 。

版本2.6中的新功能。

ctypes.get_last_error()

仅限Windows:返回LastError调用线程中系统变量的ctypes-private副本的当前值 。

版本2.6中的新功能。

ctypes.memmovedstsrccount 

与标准C memmove库函数相同:将计数字节从 src复制到dstdstsrc必须是可以转换为指针的整数或ctypes实例。

ctypes.memsetdstccount 

与标准C memset库函数相同:使用值c的计数字节填充地址dst处的内存块。dst必须是指定地址的整数或ctypes实例。

ctypes.POINTER类型

此工厂函数创建并返回新的ctypes指针类型。指针类型在内部被缓存和重用,因此重复调用此函数很便宜。type必须是ctypes类型。

ctypes.pointerobj 

此函数创建一个指向obj的新指针实例。返回的对象属于该类型POINTER(type(obj))

注意:如果您只想将指向对象的指针传递给外部函数调用,则应该使用byref(obj)哪个更快。

ctypes.resizeobjsize 

此函数调整obj的内部内存缓冲区的大小,obj必须是ctypes类型的实例。如下所示,不可能使缓冲区小于对象类型的本机大小sizeof(type(obj)),但可以放大缓冲区。

ctypes.set_conversion_mode编码错误

此函数设置ctypes对象在8位字符串和unicode字符串之间进行转换时使用的规则。 encoding必须是指定编码的字符串,如'utf-8''mbcs'错误必须是一个字符串,指定编码/解码错误的错误处理。可能的值的实例是"strict""replace",或"ignore"

set_conversion_mode()返回包含先前转换规则的2元组。在Windows上,初始转换规则在其他系统上。('mbcs', 'ignore')('ascii', 'strict')

ctypes.set_errno

errno 调用线程中系统变量的ctypes-private副本的当前值设置为value并返回先前的值。

版本2.6中的新功能。

ctypes.set_last_error

仅限Windows:将LastError调用线程中系统变量的ctypes-private副本的当前值设置 为value并返回先前的值。

版本2.6中的新功能。

ctypes.sizeofobj_or_type 

返回ctypes类型或实例内存缓冲区的大小(以字节为单位)。与C sizeof运算符相同。

ctypes.string_at地址[,大小] )

这个函数返回了从内存地址字符串地址。如果指定了size,则将其用作size,否则假定该字符串为零终止。

ctypes.WinErrorcode = Nonedescr = None 

仅限Windows:此函数可能是ctypes中最糟糕的名称。它创建了一个WindowsError实例。如果未指定代码, GetLastError则调用以确定错误代码。如果descr未指定,FormatError()则调用以获取错误的文本描述。

ctypes.wstring_at地址[,大小] )

这个函数返回了从内存地址宽字符串 地址为unicode字符串。如果指定了size,则将其用作字符串的字符数,否则假定该字符串为零终止。

   15.17.2.6。数据类型

ctypes._CData

这个非公共类是所有ctypes数据类型的公共基类。除此之外,所有ctypes类型实例都包含一个保存C兼容数据的内存块; addressof()辅助函数返回内存块的地址 。另一个实例变量暴露为 _objects; 这包含其他需要保持活动的Python对象,以防内存块包含指针。

ctypes数据类型的常用方法,这些都是类方法(确切地说,它们是元类的方法):

from_buffer来源[,偏移] )

此方法返回共享对象的缓冲区的ctypes实例 。所述对象必须支持可写缓冲器接口。可选的offset参数指定源缓冲区的偏移量(以字节为单位); 默认值为零。如果源缓冲区不够大,ValueError则会引发a。

版本2.6中的新功能。

from_buffer_copy来源[,偏移] )

此方法创建一个ctypes实例,从对象缓冲区复制缓冲区,该 缓冲区必须是可读的。可选的offset 参数指定源缓冲区的偏移量(以字节为单位); 默认值为零。如果源缓冲区不够大,ValueError则会引发a。

版本2.6中的新功能。

from_address地址

此方法使用address指定的内存返回ctypes类型实例,该内存 必须是整数。

from_paramobj 

此方法使obj适应ctypes类型。当外部函数的argtypes元组中存在类型时,使用外部函数调用中使用的实际对象调用它; 它必须返回一个可以用作函数调用参数的对象。

所有ctypes数据类型都具有此类方法的默认实现,如果是类型的实例,则通常返回obj。某些类型也接受其他对象。

in_dll图书馆名称

此方法返回由共享库导出的ctypes类型实例。name是导出数据的符号的名称,library 是加载的共享库。

ctypes数据类型的常见实例变量:

_b_base_

有时ctypes数据实例不拥有它们包含的内存块,而是共享基础对象的部分内存块。所述 _b_base_只读构件是根ctypes的对象拥有该存储器块。

_b_needsfree_

当ctypes数据实例已分配内存块本身时,此只读变量为true,否则为false。

_objects

该成员None或者是包含需要保持活动的Python对象的字典,以便内存块内容保持有效。该对象仅用于调试; 永远不要修改这本词典的内容。

   15.17.2.7。基本数据类型

ctypes._SimpleCData

这个非公共类是所有基本ctypes数据类型的基类。这里提到它是因为它包含基本ctypes数据类型的公共属性。 _SimpleCData是它的子类 _CData,因此它继承了它们的方法和属性。

在版本2.6中更改:现在可以对不包含指针但不包含指针的ctypes数据类型进行pickle。

实例具有单个属性:

value

该属性包含实例的实际值。对于整数和指针类型,它是一个整数,对于字符类型,它是单个字符串,对于字符指针类型,它是Python字符串或unicode字符串。

value属性从一个ctypes实例获取,通常是一个新的对象,每次返回。 ctypes没有实现原来的目标回报率,总是一个新的对象被创建。所有其他ctypes对象实例也是如此。

基本数据类型作为外部函数调用结果返回时,或者例如通过检索结构字段成员或数组项,将透明地转换为本机Python类型。换句话说,如果一个外国函数有一个 restypec_char_p,你总是会收到一个Python字符串, 不是一个c_char_p实例。

基本数据类型的子类不会继承此行为。因此,如果外部函数restype是其子类c_void_p,您将从函数调用中接收此子类的实例。当然,您可以通过访问value属性来获取指针的值。

这些是基本的ctypes数据类型:

ctypes.c_byte

表示C 数据类型,并将该值解释为小整数。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。signed char

ctypes.c_char

表示C char数据类型,并将该值解释为单个字符。构造函数接受可选的字符串初始值设定项,字符串的长度必须恰好是一个字符。

ctypes.c_char_p

当它指向以零结尾的字符串时表示C 数据类型。对于也可能指向二进制数据的通用字符指针, 必须使用。构造函数接受整数地址或字符串。char *POINTER(c_char)

ctypes.c_double

表示C double数据类型。构造函数接受可选的float初始化程序。

ctypes.c_longdouble

表示C 数据类型。构造函数接受可选的float初始化程序。在它是别名的平台上。long doublesizeof(long double) == sizeof(double)c_double

版本2.6中的新功能。

ctypes.c_float

表示C float数据类型。构造函数接受可选的float初始化程序。

ctypes.c_int

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。在它是别名的平台上。signed intsizeof(int) == sizeof(long)c_long

ctypes.c_int8

表示C 8位数据类型。通常是别名 。signed intc_byte

ctypes.c_int16

表示C 16位数据类型。通常是别名 。signed intc_short

ctypes.c_int32

表示C 32位数据类型。通常是别名 。signed intc_int

ctypes.c_int64

表示C 64位数据类型。通常是别名 。signed intc_longlong

ctypes.c_long

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。signed long

ctypes.c_longlong

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。signed long long

ctypes.c_short

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。signed short

ctypes.c_size_t

表示C size_t数据类型。

ctypes.c_ssize_t

表示C ssize_t数据类型。

版本2.7中的新功能。

ctypes.c_ubyte

表示C 数据类型,它将值解释为小整数。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。unsigned char

ctypes.c_uint

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。在它是别名的平台上。unsigned intsizeof(int) == sizeof(long)c_ulong

ctypes.c_uint8

表示C 8位数据类型。通常是别名 。unsigned intc_ubyte

ctypes.c_uint16

表示C 16位数据类型。通常是别名 。unsigned intc_ushort

ctypes.c_uint32

表示C 32位数据类型。通常是别名 。unsigned intc_uint

ctypes.c_uint64

表示C 64位数据类型。通常是别名 。unsigned intc_ulonglong

ctypes.c_ulong

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。unsigned long

ctypes.c_ulonglong

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。unsigned long long

ctypes.c_ushort

表示C 数据类型。构造函数接受可选的整数初始值设定项; 没有进行溢出检查。unsigned short

ctypes.c_void_p

表示C 类型。该值表示为整数。构造函数接受可选的整数初始值设定项。void *

ctypes.c_wchar

表示C wchar_t数据类型,并将该值解释为单个字符的unicode字符串。构造函数接受可选的字符串初始值设定项,字符串的长度必须恰好是一个字符。

ctypes.c_wchar_p

表示C 数据类型,该数据类型必须是指向以零结尾的宽字符串的指针。构造函数接受整数地址或字符串。wchar_t *

ctypes.c_bool

表示C bool数据类型(更准确地说,_Bool来自C99)。它的值可以是True或者False,构造函数接受任何具有真值的对象。

版本2.6中的新功能。

ctypes.HRESULT

仅限Windows:表示一个HRESULT值,其中包含函数或方法调用的成功或错误信息。

ctypes.py_object

表示C 数据类型。在没有参数的情况下调用它会创建一个指针。PyObject *NULL PyObject *

ctypes.wintypes模块提供了相当长的一段其他Windows特定的数据类型,例如HWNDWPARAMDWORD。一些有用的结构,如MSGRECT定义。

   15.17.2.8。结构化数据类型

class ctypes.Union* args** kw 

原始字节顺序的联合的抽象基类。

class ctypes.BigEndianStructure* args** kw 

大端字节顺序结构的抽象基类。

class ctypes.LittleEndianStructure* args** kw 

小端字节顺序结构的抽象基类。

具有非本机字节顺序的结构不能包含指针类型字段或包含指针类型字段的任何其他数据类型。

class ctypes.Structure* args** kw 

本机字节顺序的结构的抽象基类。

必须通过继承其中一种类型来创建具体结构和联合类型,并至少定义一个_fields_类变量。ctypes将创建描述符 s,允许通过直接属性访问来读取和写入字段。这些是

_fields_

定义结构字段的序列。项目必须是2元组或3元组。第一项是字段的名称,第二项指定字段的类型; 它可以是任何ctypes数据类型。

对于整数类型字段c_int,可以给出第三个可选项。它必须是一个小的正整数,用于定义字段的位宽。

字段名称在一个结构或联合中必须是唯一的。未选中此选项,重复名称时只能访问一个字段。

可以在定义Structure子类的类语句之后定义_fields_类变量,这允许创建直接或间接引用自身的数据类型:

  1. class List(Structure):
  2. pass
  3. List._fields_ = [("pnext", POINTER(List)),
  4. ...
  5. ]

_fields_类变量但是,必须被定义在第一次使用之前的类型(创建一个实例,sizeof()被称为其上,等等)。稍后对_fields_类变量的赋值将引发AttributeError。

可以定义结构类型的子类,它们继承基类的字段以及_fields_子子类中定义的字段(如果有的话)。

_pack_

一个可选的小整数,允许覆盖实例中结构字段的对齐方式。 _pack_必须在_fields_分配时定义,否则它将无效。

_anonymous_

一个可选序列,列出未命名(匿名)字段的名称。 _anonymous_必须在_fields_分配时定义,否则它将无效。

此变量中列出的字段必须是结构或联合类型字段。 ctypes将在结构类型中创建允许直接访问嵌套字段的描述符,而无需创建结构或联合字段。

这是一个示例类型(Windows):

  1. class _U(Union):
  2. _fields_ = [("lptdesc", POINTER(TYPEDESC)),
  3. ("lpadesc", POINTER(ARRAYDESC)),
  4. ("hreftype", HREFTYPE)]
  5. class TYPEDESC(Structure):
  6. _anonymous_ = ("u",)
  7. _fields_ = [("u", _U),
  8. ("vt", VARTYPE)]

TYPEDESC结构描述了COM数据类型,该vt字段指定哪个联合字段有效。由于该u字段被定义为匿名字段,因此现在可以直接从TYPEDESC实例访问成员。td.lptdesc并且td.u.lptdesc 是等价的,但前者更快,因为它不需要创建临时联合实例:

  1. td = TYPEDESC()
  2. td.vt = VT_PTR
  3. td.lptdesc = POINTER(some_type)
  4. td.u.lptdesc = POINTER(some_type)

可以定义结构的子类,它们继承基类的字段。如果子类定义具有单独的 _fields_变量,则在此指定的字段将附加到基类的字段中。

结构和联合构造函数接受位置和关键字参数。位置参数用于按照出现的顺序初始化成员字段_fields_。构造函数中的关键字参数被解释为属性赋值,因此它们将_fields_使用相同的名称进行初始化 ,或者为不存在的名称创建新属性_fields_

   15.17.2.9。数组和指针

class ctypes.Array* args 

数组的抽象基类。

创建具体数组类型的推荐方法是将任何ctypes数据类型与正整数相乘 。或者,您可以继承此类型并定义_length__type_类变量。可以使用标准下标和切片访问来读取和写入数组元素; 对于切片读取,所得到的物体是 本身Array

_length_

一个正整数,指定数组中元素的数量。超出范围的下标导致IndexError。将由返回len()

_type_

指定数组中每个元素的类型。

数组子类构造函数接受位置参数,用于按顺序初始化元素。

ctypes._Pointer

指针的私有抽象基类。

通过POINTER()使用将指向的类型调用来创建具体指针类型; 这是由自动完成的 pointer()

如果指针指向数组,则可以使用标准下标和切片访问来读取和写入其元素。指针对象没有大小,因此len()会提高TypeError。否定下标将在指针之前从内存中读取(如在C中),并且超出范围的下标可能会因访问冲突而崩溃(如果您很幸运)。

_type_

指定指向的类型。

contents

返回指针指向的对象。分配给此属性会将指针更改为指向指定的对象。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多