java 和C++ Socket程序发送结构体

本文介绍了在Java客户端与C++服务端通过Socket通信时,如何处理不同字节序(Java使用BIG_ENDIAN,系统通常使用LITTLE_ENDIAN)的问题。文章详细展示了Java端如何在发送和接收数据时转换结构体(如Intcmd, Intlength等)的字节序,以确保数据的正确传输。示例中,Java客户端创建报文头类并进行字节序转换,然后通过Socket发送给C++服务端,服务端接收到数据后再进行回射。" 105073721,7531054,知识图谱表示学习:理解hit@10与mean rank,"['知识图谱', '机器学习', '深度学习', '自然语言处理']
摘要由CSDN通过智能技术生成

       主要技术问题:windows,linux等系统采用LITTLE_ENDIAN字节序,而java自身采用BIG_ENGIAN字节序,BIG_ENGIAN是指低地址存放最高有效字节(MSB),而LITTLE_ENDIAN则是低地址存放最低有效字节。Java程序写的客户程序端同c++的服务端程序交互时结构体的某些数据类型需要转换字节序。本文解决方法,java客户端程序发送数据时做相应的转换字节序,等收到数据时再做一次字节序的转换。

       现在的网络程序多数采用可靠交付的TCP协议,其采用字节流的传输方式,c++程序中用结构体来模拟报头以此界定每次发送的报文。所以网络中整个字节流的格式:报头+数据负载+报头+数据负载……

      本文基于c++写服务端和java写客户端应用场景描述。服务端程序收到客户端发送的报文后将原报文做一个回射。

采用报头:

Struct Header{

         Intcmd;            //标示报文用途

         Intlength;        //数据负载长度

         Intpara1;                   //服务端处理完结果

Int para2;                   //服务端处理完结果

Int para3;                   //服务端处理完结果

}; 此时套接口的读写方式为先读报头,在报头中取出数据负载的长度,然后再读相应字节的数据。

        

Java客户端代码:

报头类:

publicclassMsgHeader {

    privateintcmd;

    privateintlength;

    privateintpara1;

    privateintpara2;

    publicint getCmd() {

        returncmd;

    }

    publicvoid setCmd(int cmd) {

        this.cmd = cmd;

    }

    publicint getLength() {

        returnlength;

    }

    publicvoid setLength(int length) {

        this.length = length;

    }

    publicint getPara1() {

        returnpara1;

    }

    publicvoid setPara1(int para1) {

        this.para1 = para1;

    }

    publicint getPara2() {

        returnpara2;

    }

    publicvoid setPara2(int para2) {

        this.para2 = para2;

    }

    publicint getPara3() {

        returnpara3;

    }

    public

在C中,我们可以使用结构体来表示复合数据类型,例如: ```c struct person { char name[50]; int age; float height; }; ``` 在Java中,我们可以使用类来表示类似的复合数据类型,例如: ```java public class Person { public String name; public int age; public float height; } ``` 要在C和Java之间进行结构体转换,我们可以使用序列化和反序列化技术。在C中,我们可以使用标准库中的函数将结构体转换为字节数组,例如: ```c struct person p = {"John", 30, 1.75}; unsigned char buffer[100]; memcpy(buffer, &p, sizeof(struct person)); ``` 在Java中,我们可以使用Java序列化技术将对象转换为字节数组,例如: ```java Person p = new Person(); p.name = "John"; p.age = 30; p.height = 1.75f; ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(baos); oos.writeObject(p); byte[] buffer = baos.toByteArray(); ``` 要在C和Java之间进行通信,我们可以使用Socket API。在C中,我们可以使用标准库中的函数创建和连接Socket,例如: ```c int sockfd = socket(AF_INET, SOCK_STREAM, 0); struct sockaddr_in servaddr; memset(&servaddr, 0, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); servaddr.sin_port = htons(8080); connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr)); ``` 在Java中,我们可以使用Java Socket API创建和连接Socket,例如: ```java Socket socket = new Socket("127.0.0.1", 8080); ``` 在C中,我们可以使用标准库中的函数发送和接收数据,例如: ```c send(sockfd, buffer, sizeof(buffer), 0); recv(sockfd, buffer, sizeof(buffer), 0); ``` 在Java中,我们可以使用Java Socket API发送和接收数据,例如: ```java OutputStream os = socket.getOutputStream(); os.write(buffer); InputStream is = socket.getInputStream(); is.read(buffer); ``` 要在C和Java之间进行结构体转换和通信,我们可以将上述技术结合起来,例如: 在C中,我们可以将结构体转换为字节数组,然后发送Java端: ```c struct person p = {"John", 30, 1.75}; unsigned char buffer[100]; memcpy(buffer, &p, sizeof(struct person)); send(sockfd, buffer, sizeof(buffer), 0); ``` 在Java中,我们可以接收字节数组,然后将其反序列化为Java对象: ```java InputStream is = socket.getInputStream(); byte[] buffer = new byte[100]; is.read(buffer); ByteArrayInputStream bais = new ByteArrayInputStream(buffer); ObjectInputStream ois = new ObjectInputStream(bais); Person p = (Person)ois.readObject(); ``` 然后,我们可以在Java端将Java对象转换为字节数组,然后发送给C端: ```java ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(baos); oos.writeObject(p); byte[] buffer = baos.toByteArray(); OutputStream os = socket.getOutputStream(); os.write(buffer); ``` 在C端,我们可以接收字节数组,然后将其转换为C结构体: ```c unsigned char buffer[100]; recv(sockfd, buffer, sizeof(buffer), 0); struct person p; memcpy(&p, buffer, sizeof(struct person)); ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值