如何将const char *从python传递给c函数

时间:2016-06-22 11:15:32

标签: python ctypes python-c-api

我在Python中使用ctypes来打开一个用C ++编写的文件。

我的C ++代码:

extern "C" {
void openfile(const char *filename) {
    cout<<"File to open for writing = " <<filename<<endl;
    FILE *fp = fopen(filename,"w");
    fprintf(fp,"writing into file");
    fclose(fp);
}
}

我的Python代码:

>>> import ctypes
>>> lib = ctypes.cdll.LoadLibrary('/in/vrtime/mahesh/blue/rnd/software/test/test.so')
>>> outfile = "myfirstfile.txt"
>>> lib.openfile(outfile)
File to open for writing = m

我的文件名为m,这是我文件的第一个char字符。

如何将整个字符串传递给C端?

1 个答案:

答案 0 :(得分:10)

在python3中(你肯定在python2上使用python3,你的代码很幸运)  字符串存储为wchar_t[]缓冲区,因此当您传递"myfirstfile.txt"时 C函数将其arg视为"m\0y\0...",这显然是长度为1的C字符串。 这是表现出来的问题:

In [19]: from ctypes import cdll, c_char_p

In [20]: libc = cdll.LoadLibrary("libc.so.6")

In [21]: puts = libc.puts

In [22]: puts('abc')
a

您应该将bytes对象

传递给C函数
In [23]: puts(b'abc')
abc

您可以将str转换为bytes,如下所示:

puts(my_var.encode())

为避免进一步混淆,您可以指定C函数的参数类型:

In [27]: puts.argtypes = [c_char_p]

现在该函数接受bytes(ctypes将其转换为char*):

In [28]: puts(b'abc')
abc

但不是str

In [30]: puts('abc')
---------------------------------------------------------------------------
ArgumentError                             Traceback (most recent call last)
<ipython-input-26-aaa5b59630e2> in <module>()
----> 1 puts('abc')

ArgumentError: argument 1: <class 'TypeError'>: wrong type