优秀的编程知识分享平台

网站首页 > 技术文章 正文

C/C++实战014:字符串转换及Python传参数组

nanyue 2024-08-06 18:05:07 技术文章 6 ℃

今天在写北京精雕宏变量获取的时候遇到一个传值为数组的例子,在获取宏参数时需要我们提供获取值的个数和对应的#代码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()
最近发表
标签列表