ctypes c_char_p的不同行为?

Posted

技术标签:

【中文标题】ctypes c_char_p的不同行为?【英文标题】:Different behaviour of ctypes c_char_p? 【发布时间】:2014-07-14 04:13:14 【问题描述】:

我对不同版本的 python 的这种行为感到困惑,不明白为什么?

Python 2.7.5 (default, Aug 25 2013, 00:04:04) 
[GCC 4.2.1 Compatible Apple LLVM 5.0 (clang-500.0.68)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> c="hello"
>>> a=ctypes.c_char_p(c)
>>> print(a.value) 
hello

Python 3.3.5 (default, Mar 11 2014, 15:08:59) 
[GCC 4.2.1 Compatible Apple LLVM 5.0 (clang-500.2.79)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> c="hello" 
>>> a=ctypes.c_char_p(c)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: bytes or integer address expected instead of str instance

一个工作,而另一个给我一个错误。哪一个是正确的?

如果它们都正确,我怎样才能在 3.3.5 中实现与 2.7 相同的行为?我想将 char 指针从 python 传递给 C。

【问题讨论】:

在 Python 3 中使用bytes,即c = b"hello"c_char_p 实例指向 bytes 对象的私有缓冲区,因此仅将其用于不会修改字符串的 const 参数。 【参考方案1】:

c_char_p_SimpleCData 的子类,具有_type_ == 'z'__init__ 方法调用类型的setfunc,对于简单类型'z'z_set

在 Python 2 中,z_set function (2.7.7) 用于处理 strunicode 字符串。在 Python 3 之前,str 是一个 8 位字符串。 CPython 2.x str 内部使用 C 空终止字符串(即以 \0 终止的字节数组),z_set 可以为此调用 PyString_AS_STRING(即获取指向 @ 的内部缓冲区的指针987654342@ 对象)。 unicode 字符串首先需要编码为字节字符串。 z_set 自动处理此编码,并在 _objects 属性中保留对编码字符串的引用。

>>> c = u'spam'
>>> a = c_char_p(c)
>>> a._objects
'spam'
>>> type(a._objects)
<type 'str'>

在 Windows 上,默认 ctypes 字符串编码为 'mbcs',错误处理设置为 'ignore'。在所有其他平台上,默认编码为'ascii',带有'strict' 错误处理。要修改默认值,请致电ctypes.set_conversion_mode。例如,set_conversion_mode('utf-8', 'strict')

在 Python 3 中,z_set function (3.4.1) 不会自动将 str(现在是 Unicode)转换为 bytes。 Python 3 中的范式转变为严格区分字符串和二进制数据。删除了 ctypes 默认转换,函数 set_conversion_mode 也是如此。你必须传递c_char_p 一个bytes 对象(例如b'spam''spam'.encode('utf-8'))。在 CPython 3.x 中,z_set 调用 C-API 函数 PyBytes_AsString 以获取指向 bytes 对象的内部缓冲区的指针。

请注意,如果 C 函数修改了字符串,那么您需要改为使用 create_string_buffer 创建一个 c_char 数组。查找要键入为const 的参数以知道使用c_char_p 是安全的。

【讨论】:

感谢您的详细解答。 总之,调用 ctypes.c_char_p(my_string.encode('utf-8')) 让它在 Python 3.x 中工作。

以上是关于ctypes c_char_p的不同行为?的主要内容,如果未能解决你的问题,请参考以下文章

如何像 python ctypes 那样指定 JNR 指针

C DLL 和 Python CTypes - 发布/调试中的不同返回

ctypes 模块

使用ctypes实现python类型和C语言类型之间的相互转化

求助python调用dll函数里,关于数据指针和长度应该如何获得呢?

转换与 CType