C++与Python之间跨进程通信(socket实现)

C++与Python之间跨进程通信(socket实现)

1.引言

之前写过一篇Python调用C++程序的实现方法,这里相反,希望使用Python协助C++完成某些任务。一种解决思路为实现RPC调用,使用C++端(以下称客户端)发送数据,Python端(以下称服务端)处理数据并返回的方法,进一步来说,转换为C++与Python之间通信的问题。

2.实现思路

因为客户端可能希望使用的函数多种多样,这里为了保证灵活,服务端与客户端均写成框架的形式(取名CPhone),在实际使用的过程中,可以方便服务端即时自定义函数,客户端灵活调用。服务端文件结构如下:
在这里插入图片描述
包CPhone为服务器框架,在实际使用的时候只需编写test.py文件中的内容(定义函数)即可。
客户端:
在这里插入图片描述
使用的时候,通过callPyFun()函数调用即可。
为了统一数据格式,在服务端使用numpy的ndarray数据类型,在客户端使用opencv的mat数据类型。

3. 具体代码

(1)Python服务端

CPhone/__init__.py

import numpy as np
import socket
from datetime import datetime
from threading import Thread
import time
import os
from operator import methodcaller


class CPhone:
    def __init__(self):
        self.g_conn_pool = []
        self.server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        self.thread = Thread(target=self._accept_client) #主线程
        self.thread.setDaemon(True)

    def run(self,ip="localhost",port=8888,max_n=0):
        self.server.bind((ip, port))
        self.server.listen(max_n)
        self.thread.start()

        log = """--------------------------\n--------------------------
%s\t\n(ip=%s,port=%d)\n\tstart listening...
"""% (str(datetime.now()), ip, port)
        if os.path.exists('./log.txt'):
            with open('./log.txt','a',encoding='utf-8') as fo:
                fo.write(log)
        else:
            with open('./log.txt','w',encoding='utf-8') as fo:
                fo.write(log)
        print(log)
        while True:
            cmd = input("""--------------------------
    输入1:查看当前在线人数
    输入0:关闭服务器
--------------------------
""")
            if cmd == '1':
                print("--------------------------")
                print("当前在线人数:\n", len(self.g_conn_pool))
            elif cmd == '0':
                exit()

    def _data_handle(self,client,address): #此函数主要用于解析数据,调用函数处理,并将结构序列化为字符串
        func_name,mat = self._recvall(client)
        #cv2.imwrite('xx.jpg', mat)

        return_data = methodcaller(func_name,mat)(self)

        shape_str = [str(i) for i in return_data.shape]
        shape_str = ','.join(shape_str)
        data_str = str(return_data.reshape(-1).tolist()).replace(' ', '').replace('[', ',').replace(']', ',')
        shape_str = ',%s,%d,' % (shape_str, len(data_str))
        self._sendall(client, shape_str, data_str)
        time.sleep(2)
        client.close()
        self.g_conn_pool.remove(client)
        log = ('%s\t用户(ip:%s port:%d)\t已下线\n'% (str(datetime.now()),address[0], address[1]))
        with open('./log.txt', 'a',encoding='utf-8') as fo:
            fo.write(log)
        print(log)
'''
为了解决C++栈溢出的问题,在发送和接受数据的时候均循环传输。
发送数据时先发送头信息,主要储存数据的shape,调用的函数名等。
'''

    def _recvall(self,sock):
        header = sock.recv(20)
        header = str(header).split('\\',1)[0].replace("b'",'')
        func_name,channels,height,width = header.split(',')
        channels,height,width = int(channels),int(height),int(width)
        count = channels*height*width
        buf = b''
        while count:
            newbuf = sock.recv(count)
            if not newbuf: return None
            buf += newbuf
            count -= len(newbuf)
        data = np.frombuffer(buf, dtype='uint8')
        mat = data.reshape(height, width, channels)
        return func_name,mat

    def _sendall(self,sock,header_str,data_str):
        sock.send(bytes(header_str, encoding="ascii"))
        data_bt = bytes(data_str, encoding="ascii")
        total = len(data_str)
        sended = 0
        while sended<total:
            send = min(total-sended,128*128*3)
            sock.send(data_bt[sended:sended+send])
            sended += send

    def _accept_client(self): #此函数用于接收客户端的链接请求,并为其开启一个线程
        while True:
            client, address = self.server.accept()  # 阻塞,等待客户端连接
            # 加入连接池
            self.g_conn_pool.append(client)
            log = ('%s\t用户(ip:%s port:%d)\t已上线\n' % (str(datetime.now()), address[0], address[1]))
            with open('./log.txt','a',encoding='utf-8') as fo:
                fo.write(log)
            print(log)
            # 给每个客户端创建一个独立的线程进行管理
            thread = Thread(target=self._data_handle, args=(client,address))
            # 设置成守护线程
            thread.setDaemon(True)
            thread.start()


实际使用过程中,需进行如下操作
test.py

from CPhone import CPhone
import numpy as np
import torchvision.transforms as transforms


class MyCPhone(CPhone): #继承CPhone
    def normalize(self,mat): #这里定义一个归一化函数测试
        transform = transforms.Compose([
            transforms.ToTensor(),
            transforms.Normalize(
                mean=np.array([0.5, 0.5, 0.5]),
                std=np.array([0.5, 0.5, 0.5])),
        ])
        remat = transform(mat).numpy()
        return remat

    def mean(self,mat): #这里定义一个求均值函数测试
        remat = np.mean(mat)
        remat = np.array([remat])
        return remat


if __name__ == '__main__':
    test = MyCPhone()
    test.run() #开启服务

(2)C++客户端

client.cpp

#include <stdio.h>  
#include <Winsock2.h>  
#pragma comment(lib,"ws2_32.lib")
#pragma warning(disable:4996)
#include <iostream>
#include <opencv.hpp>
#include <opencv2/core.hpp>
#include <opencv2/imgproc.hpp>
#include <iostream>
#include <cmath>
#include <vector>
#define BuffSize 1024
#define MaxReceive 256*256*3

class CPhone
{
private:
	WORD wVersionRequested;
	WSADATA wsaData;
	int err;
	SOCKET sockClient;
	SOCKADDR_IN addrSrv;
	int sec;
public:
	CPhone(const char*,int);
	cv::Mat callPyFun(const char*, cv::Mat);
	cv::Mat createMatFromImage(const char*);
};

CPhone::CPhone(const char* ip="127.0.0.1",int port=8888)
{
	this->wVersionRequested = MAKEWORD(1, 1);

	this->err = WSAStartup(this->wVersionRequested, &(this->wsaData));
	if (this->err != 0) {
		throw "error!";
	}

	if (LOBYTE(this->wsaData.wVersion) != 1 ||
		HIBYTE(this->wsaData.wVersion) != 1) {
		WSACleanup();
		throw "error!";
	}
	this->sockClient = socket(AF_INET, SOCK_STREAM, 0);
	this->addrSrv.sin_addr.S_un.S_addr = inet_addr(ip);
	this->addrSrv.sin_family = AF_INET;
	this->addrSrv.sin_port = htons(port);
	this->sec = connect(this->sockClient, (SOCKADDR*)&this->addrSrv, sizeof(SOCKADDR));
	if (SOCKET_ERROR == this->sec)
	{
		throw "error!";
	}
}


cv::Mat CPhone::callPyFun(const char* funName, cv::Mat mat)
{
	char* pos = (char*)mat.data;
	int colByte = mat.cols * mat.channels() * sizeof(uchar);
	//cout << mat.size[0];
	char header[30];
	//memcpy(header, &data[], 15);  //buf为储存复制内容的目标数组,sed 为字节数
	sprintf(header, "%s,%d,%d,%d",funName,mat.channels(),mat.size[0],mat.size[1]);
	send(this->sockClient, header, 30, 0);
	for (int i = 0; i < mat.rows; i++)
	{
		char* data = (char*)mat.ptr<uchar>(i); //第i行首地址
		int sedNum = 0;
		char buf[BuffSize] = { 0 };

		//按行拆分,发送数据
		while (sedNum < colByte)
		{
			int sed = BuffSize<(colByte - sedNum)?BuffSize: (colByte - sedNum);
			memcpy(buf, &data[sedNum], sed);  //buf为储存复制内容的目标数组,sed 为字节数
			int SendSize = send(this->sockClient, buf, sed, 0);
			if (SOCKET_ERROR == SendSize)
			{
				cv::Mat mat;
				return  mat;
			}
			/*else
			cout << SendSize << endl;*/
			sedNum += SendSize;
		}
	}
	char shape_str[40]; //收到的shape的字符串
	
	int ns1 = recv(this->sockClient, shape_str,40, 0); //收到的字符串的长度
	shape_str[ns1] = '\0';
	int shape[6] = {0,0,0,0,0,0}; //shape的值
	char temp_num[40];  //当前储存的数字字符
	int isnum = 0; //当前储存了几个字符
	for (int i = 0; i < ns1; i++)
	{
		if (shape_str[i]==',')
		{
			if (isnum) // 已经储存了字符
			{
				temp_num[isnum] = '\0';
				shape[shape[0]+1] = atoi(temp_num);
				shape[0]++;
				isnum = 0;
			}
			else //开始的时候没有储存字符
			{
				continue;
			}
		}
		else //数字字符
		{
			temp_num[isnum] = shape_str[i];
			isnum++;
		}
	}
	int n_num = 1;
	for (int i = 0; i < shape[0]-1; i++)
	{
		n_num *= shape[i + 1];
	}
	int str_len = shape[shape[0]];
	std::vector<double> vet;
	isnum = 0;  //已经储存的字符
	int ns2;
	while (str_len > 0) //循环接收
	{
		char* recvBuf = new char[MaxReceive];//收到的数据
		ns2 = recv(this->sockClient, recvBuf, MaxReceive, 0); //返回字符串的长度
		str_len -= ns2;
		recvBuf[ns2] = '\0';
		for (int i = 0; i < ns2; i++)
		{
			if (recvBuf[i] == ',') //如果遇到','
			{
				if (isnum) //如果已经储存了字符
				{
					temp_num[isnum] = '\0';
					vet.push_back(atof(temp_num));
					isnum = 0;
				}
				else //此时为开始的时候
				{
					continue;
				}
			}
			else
			{
				temp_num[isnum] = recvBuf[i];
				isnum++;
			}
		}
	}
	cv::Mat return_mat = cv::Mat(vet,true);
	switch (shape[0]-1)
	{
	case 1:
		return return_mat.reshape(shape[1]).clone();
		break;
	case 2:
		return return_mat.reshape(shape[1],shape[2]).clone();
		break;
	case 3:
		return return_mat.reshape(shape[3],(shape[1], shape[2])).clone();
		break;
	default:
		cv::Mat mat;
		return  mat;
		break;
	}
}

cv::Mat CPhone::createMatFromImage(const char *path)
{
	return cv::imread(path, cv::IMREAD_UNCHANGED);
}



int main()
{
	CPhone phone;
	cv::Mat img = phone.createMatFromImage("C:\\Users\\Lenovo\\Desktop\\xx.jpg"); //这里读取一张3*640*640的图片
	cv::Mat remat;
	remat = phone.callPyFun("normalize",img).clone(); //参数为调用的函数名,mat数据
	std::cout << remat;
	return 0;
}

使用的时候只需编写main()函数里的代码即可调用。
以下为对两个函数调用的返回结果
mean
在这里插入图片描述

normalize:
在这里插入图片描述

  • 5
    点赞
  • 36
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 基于TCP的socket网络传输视频(C++, python) 可以实现C++ to C++Python to pythonC++ to Python的视频或图像传输。 ### 一. 概述 ### Socket的英文原义是“孔”或“插座”。作为BSD UNIX的进程通信机制,取后一种意思。通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,可以用来实现不同虚拟机或不同计算机之间的通信。在Internet上的主机一般运行了多个服务软件,同时提供几种服务。每种服务都打开一个Socket,并绑定到一个端口上,不同的端口对应于不同的服务。Socket正如其英文原意那样,像一个多孔插座。一台主机犹如布满各种插座的房间,每个插座有一个编号,有的插座提供220伏交流电, 有的提供110伏交流电,有的则提供有线电视节目。 客户软件将插头插到不同编号的插座,就可以得到不同的服务。网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个socket。 而socketsocket之间的连接以及数据传输需要一种规则,也就是我们通常所说的网络传输协议,最常用的有TCP和UDP,这两种协议的区别如下: 1.基于连接与无连接; 2.对系统资源的要求(TCP较多,UDP少); 3.UDP程序结构较简单; 4.流模式与数据报模式 ; 5.TCP保证数据正确性,UDP可能丢包,TCP保证数据顺序,UDP不保证。 接下来将以图片传输为例,用PythonC++实现服务端和客户端。这里不用语言得到的端口之间也可以互相连接。 ### 二. 运行要求 ### (1)OpenCV (2)Python 2.7 (3)C++的源文件要求windows环境 ### 三. 参考资料 ### -------- 该资源内项目源码是个人的毕设,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! <项目介绍> 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。 --------
### 回答1: 实现网络通信监控需要通过以下几个步骤来实现: 1. 监听网络通信:采用代理服务器的方式,将网络流量导向到一个中间设备上,然后在该设备上监测网络通信。可以使用网络抓包工具,如Wireshark,来捕获并分析网络数据包。 2. 分析网络数据:对捕获到的网络数据包进行解析和分析,提取关键信息,如源地址、目标地址、协议类型、端口号等。可以采用数据包解析库,如Scapy或Pcap来辅助分析。 3. 实时监控:将分析得到的网络信息进行处理,并实时监控网络数据的流向和传输状态。可以使用编程语言,如Python或Java,利用网络编程库,如socket或scoket.io来开发监控程序。 4. 报警机制:当监测到异常或危险情况时,通过邮件、短信或弹窗等方式进行即时报警通知。可以设置关键指标的阈值,并通过条件判断来触发报警。 5. 数据记录与分析:将监测到的网络数据进行记录,并可以对其进行分析和统计,以便后续分析网络流量、检测网络攻击或优化网络性能。 6. 可视化展示:将监测到的网络数据以可视化的方式展示在监控界面上,可以使用图表、表格、地图等形式展示相关信息,方便用户查看和理解。 总之,实现网络通信监控需要对网络数据进行分析和处理,并利用报警机制和数据分析来提升网络安全性和性能。同时,合理的可视化展示可以帮助用户直观地了解网络通信的情况。 ### 回答2: C实现网络通信监控可以通过以下步骤实现。 首先,需要使用C语言的网络编程库,例如sockets库,以便与网络通信进行交互。可以使用socket()函数创建一个套接字,并使用bind()函数将其绑定到特定的IP地址和端口号上。 接下来,可以使用listen()函数将套接字设置为监听状态,以便接受传入的网络连接。可以创建一个循环,使用accept()函数接受连接请求,并为每个连接创建一个新的线程或进程进行处理。 在每个连接中,可以使用read()函数从套接字中读取数据,或使用write()函数将数据写入套接字中。可以根据需要制定通信协议和数据格式。 为了实现网络通信的监控,可以添加适当的监控机制。例如,可以使用select()函数来监视套接字的状态变化,以便在有新的数据可读或写入时及时做出相应的处理。 此外,还可以使用日志记录或打印输出等方式,将网络通信过程中的各个步骤和数据进行记录,以便进行监控和故障排除。可以使用文件操作函数如fprintf()来写入日志文件,或使用printf()函数将日志信息打印到控制台。 最后,为了确保网络通信监控的稳定性和可靠性,还可以添加错误处理机制。例如,可以检测套接字连接的异常或错误,及时关闭连接并释放资源,以避免潜在的资源泄露或崩溃。 总之,通过以上步骤和技术,可以使用C语言实现网络通信监控,并对网络通信进行有效地监控和管理。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值