python数字串怎么取出(CC实战014)
python数字串怎么取出(CC实战014)char* test(int VarNo[64] int len) { try{ CString val=""; for(int i=0;i<len;i ){ CString buf; buf.Format("%d" VarNo[i]); buf = " "; //分隔符 val = buf; //自加拼接字符串 } char* res=new char[]; strcpy(res val.GetBuffer(val.GetLength())); return res; }catch(...){ return 0; } } memcpy内存区拷贝除了strcpy方法我们还可以使用memcpy方法,memcpy与strcpy不同之处在于strcpy只能复制字符串,而memcpy可以复制任意内容,strcpy不需要指定长
今天在写北京精雕宏变量获取的时候遇到一个传值为数组的例子,在获取宏参数时需要我们提供获取值的个数和对应的#代码int数组(最多一次可传递64个),接口返回给我们的是一个double数组(最多一次可返回64个),在c 我们只需定义一个数组然后传给方法即可,但今天要做的是通过python来调用c 的动态连接库。
python如何传递int数组ctypes是Python的一个外部库,提供和C语言兼容的数据类型,通过ctypes可以调用c 编写的DLL中的接口函数,所以这里我们用ctypes来定义与c类型对应的数组类型,我在python中写了一个测试方法来调用c 的接口函数。
def dome(): dll = ctypes.cdll.LoadLibrary('C:/Users/Java/Desktop/JD/Debug/JD.dll')#引用c 动态链接库 num=7 #定义数组长度 IntArray = ctypes.c_int * num #申明一个数组 ia = IntArray(501 502 503 504 505 506 507) #给数组赋值 res = dll.test(ia num)#调用c 动态链接库接口 pyResult=ctypes.string_at(res) #转换c 返回值 print(pyResult)#输出结果 dome() C 接受传参
由于精雕接口需要连接精雕才能实现,所以这里我写了一个测试方法来验证是否能够成功获取到Python传过来的值,然后我们通过循环遍历再将接收到的值提取出来转成char*返回给Python。
为什么不直接返回double数组?因为在c/c 中没有数组这种基本数据类型,所以无法直接返回一个数组,因此这里我采取 的是将数组转成字符串格式输出。
CString转char*问题GetBuffer字符转换
这里遇到个转类型的问题,那就是CString转char*,这里我先用的是GetBuffer,将CString转char*并在c中对转换后的值进行打印,输出结果无误,而当Python调用时输出的结果却是以十六进制数 dd 规定的字符: \xdd \xdd \xdd \xdd \xdd \xdd \xdd ......
char* test(int VarNo[64] int len) { try{ CString val=""; for(int i=0;i<len;i ){ CString buf; buf.Format("%d" VarNo[i]); buf = " "; //分隔符 val = buf; //自加拼接字符串 } int len=val.GetLength(); char* res=val.GetBuffer(len); //将CString转char* printf("res=%s\n" res); //打印res值 return res; }catch(...){ return 0; } } strcat字符串追加
换个思路,直接转不行我就新建一个char* res空对象,然后将val追加到res中去。这里用到的strcat方法,将val 所指向的字符串追加到 res 所指向的字符串的结尾,python打印的结果为:pyResult= b'\xfd\xfd\xfd\xfd501 502 503 504 505 506 507 '。正确的结果是出来了,但是前面多了些字符:xfd\xfd\xfd\xfd
char* test(int VarNo[64] int len) { try{ CString val=""; for(int i=0;i<len;i ){ CString buf; buf.Format("%d" VarNo[i]); buf = " "; //分隔符 val = buf; //自加拼接字符串 } char* res=new char[]; strcat(res val.GetBuffer(val.GetLength())); return res; }catch(...){ return 0; } } strcpy字符串复制
追加失败我们来尝试下使用strcpy方法,同样新建一个char* res空对象,然后将val复制到res中去。python打印的结果为:pyResult= b'501 502 503 504 505 506 507 ',刚好是我们想要的结果。
char* test(int VarNo[64] int len) { try{ CString val=""; for(int i=0;i<len;i ){ CString buf; buf.Format("%d" VarNo[i]); buf = " "; //分隔符 val = buf; //自加拼接字符串 } char* res=new char[]; strcpy(res val.GetBuffer(val.GetLength())); return res; }catch(...){ return 0; } } memcpy内存区拷贝
除了strcpy方法我们还可以使用memcpy方法,memcpy与strcpy不同之处在于strcpy只能复制字符串,而memcpy可以复制任意内容,strcpy不需要指定长度,遇到结束符"\0"才结束,所以容易溢出。memcpy则是根据其第3个参数决定复制的长度。所以我们也可以用memcpy来实现以上功能,将从存储区val的值 复制 n 个字符到存储区 res,python打印的结果为:pyResult= b'501 502 503 504 505 506 507 ',同样是我们想要的结果。
char* test(int VarNo[64] int len) { try{ CString val=""; for(int i=0;i<len;i ){ CString buf; buf.Format("%d" VarNo[i]); buf = " "; //分隔符 val = buf; //自加拼接字符串 } char* res=new char[]; memcpy(res val val.GetLength() 1); //长度需 1,不然结尾会多一个\xb6字符 return res; }catch(...){ return 0; } }
要是实在觉得传数组麻烦,就直接传字符串给c ,让c 来处理字符串,提取需要的内容再转成相应的数组类型,返回的时候也直接用字符串形式传递,Python拿到结果后自行拆解,得到需要的数据即可,总之怎么方便怎么来。
def dome(): dll = ctypes.cdll.LoadLibrary('C:/Users/Java/Desktop/JD/Debug/JD.dll') array = '501 502 503 504' VarNo=bytes(array encoding = 'utf-8') res = dll.test(VarNo) pyResult=ctypes.string_at(res) print('pyResult=' pyResult) dome()