python与C++交互

时间:2025-05-11 16:37:26

python和C++能进行有效的交互,c++调用Python的一些小用法

写了一个python脚本导入发生异常,可能是编码问题(如存在中文),
Python默认的是ASCII
可加上:
#!/usr/bin/python
# -*- coding: utf-8 -*-
参见:https://www.python.org/dev/peps/pep-0263/

定义类C数据结构:
class Point(Structure):
_pack_ = 1
_fields_ = [('x', c_uint8),
('y', c_uint8),
]
定义点数组初始化:
pointList = [Point(0,0),
Point(1,1),
]
访问:pointList[i].x, pointList[i].y
也可强制转换为指针访问
pPoint = cast(addressof(point),POINTER(Point))
指针有一个属性contents为实例对象
pPoint.contents.x, pPoint.contents.y

想要定义数组:
class Array(Structure):
_pack_ = 1
_fields_ = [('iArray', c_uint8 * 5),
('bArray', c_ubyte * 5),
]
定义缓冲区:
buf = c_buffer("Hello",520)
buf = create_string_buffer("Hello", 10)
第二个是新一点的,旧的也可以用

ctypes的数据类型:https://docs.python.org/2/library/ctypes.html#ctypes.c_ubyte

addressof(obj): 获取ctypes类型地址
byref(obj, offset):
返回ctypes类型轻量级指针
对应以下C代码:(((char *)&obj) + offset)
sizeof: 计算变量和类型的字节数

调用c库的一些函数:
libc = cdll.msvcrt
fopen = libc.fopen
fwrite = libc.fwrite
fclose = libc.fclose
fseek = libc.fseek
fread = libc.fread
ftell = libc.ftell
memset = libc.memset
memcpy = libc.memcpy
可获取函数地址,调用与C一样,注意这些函数操作需要地址
可以用addressof获取ctypes类型的地址

接收C++传来的参数:
如果传过来的是Unicode字符需要转换为python字符串,否则会出现意想不到的错误
虽然用打印和写入文件的方式写入param内容是正确的,但是实际调用libc库时仍会
找不到路径
str(unicode(param))

使用python时如果更加了解其内在机制,会对使用该语言更有帮助

下面上代码:

 #!/usr/bin/python
# -*- coding: utf-8 -*-
import os, sys
from ctypes import * class CPoint(Structure):
_pack_ = 1
_fields_ = [('x', c_uint8),
('y', c_uint8),
] class CArray(Structure):
_pack_ = 1
_fields_ = [('iArray', c_uint8 * 5),
('bArray', c_char * 5),
] def EditBuf(buf, len):
tmp = buf
i = 0
val = c_char('')
while i < len:
memmove(addressof(tmp)+i,addressof(val),sizeof(val))#addressof(tmp)+i相当于指针移位
i += 1 libc = cdll.msvcrt
fopen = libc.fopen
fwrite = libc.fwrite
fclose = libc.fclose
fseek = libc.fseek
fread = libc.fread
ftell = libc.ftell
memset = libc.memset
memcpy = libc.memcpy if __name__ == '__main__':
point = CPoint(5,6)
pPoint = cast(addressof(point),POINTER(CPoint))
print "point.x = %d , point.y = %d" %(point.x, point.y)
print "pPoint.x = %d , pPoint.y = %d" %(pPoint.contents.x, pPoint.contents.y)
print point, pPoint.contents#这里两个地址值不一样说明系统为pPoint新申请了一个Point保存复制的值 pointList = [CPoint(0,1),
CPoint(2,3)
]
print pointList[0].x, pointList[0].y#数组 iArr = (c_uint8 * 5)(5,6,7,8)
bArr = (c_char * 5)('a','b','c')
print bArr[:]
cArray = CArray(iArr, bArr[:])#c_uint8和c_char赋初始值的方式不同,具体还没研究
print cArray.iArray[:]
print cArray.bArray[:] buf = c_buffer("",8)#类似c_char*8
print buf.value
EditBuf(buf,8)#修改buf值,对内存修改
print buf.value
memmove(addressof(buf),byref(c_char('a')),sizeof(c_char))
print buf.value #C库函数调用
fp =fopen("D:\\pythonCallClib.txt","wb+")
fwrite(buf,1,sizeof(buf),fp)#
fseek(fp,3,0)
fwrite(addressof(c_char('')),1,1,fp)
fclose(fp) #sizeof
print "c_uint8 = %d bytes, c_int = %d bytes, buf = %d bytes" %(sizeof(c_uint8), sizeof(c_int), sizeof(buf)) #cast 可以将buf强制转换成我们自定义的类型
buf = create_string_buffer("abc",8)
p = cast(addressof(buf),POINTER(CPoint))
print p.contents.x, p.contents.y

python与C++交互

python与C++交互