Java 与 C socket通信传输结构体消息/Java解析C语言结构体

Linux C与java实现 socket通信,网上部分例子通信消息格式都是传递的字节流消息,直接传递char型数组消息。而在网络协议、通信控制、嵌入式系统、驱动开发等地方,我们经常要传送的不是简单的字节流(char型数组),而是多种数据组合起来的一个整体,其表现形式是一个结构体。这时候传输的内容如果保存在char型数组中,编程复杂,易出错,而且一旦控制方式及通信协议有所变化,程序就要进行非常细致的修改,非常容易出错。这个时候只需要一个结构体就能搞定。当然,如果是都是C语言编写的client和server传输结构体,解析起来很简单。只要两端定义的结构体形式一样,就不会出问题。而Java里面没有结构体,要怎么才能正确解析出来结构体的数组呢?下面直接上代码看看.

下面的代码是我在Linux C测试的,结构体包含了几种基本类型变量,我们用Java解析出来结构体中各个基本类型的数据。

Linux C发送结构体/ client端

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <unistd.h>
/**
* Linux C socket UDP通信,传输结构体。2017/8/20
*@author Lian
*/
#include <stdint.h>

typedef struct{
    uint8_t lxx;
    int nodeID;
    int updateFlg;
    uint16_t slot;
    uint32_t slotBitmap;
    float BitMap;
} test_stru,*p_stru;

int main(){
    test_stru test1 = {200,121,127,4,4294967279,5648.525};//结构体初始化
    printf("test_stru size is  %d\n",sizeof(test_stru));
    p_stru p_test = &test1;
    size_t inlen = sizeof(test_stru);
    char outbuf[1024]={};
    size_t outlen = sizeof(outbuf);
    int brdcFd;
    if((brdcFd=socket(PF_INET,SOCK_DGRAM,0))<0){
        printf("socket fail\n");
        return -1;
    }
    int optval =1;
    setsockopt(brdcFd,SOL_SOCKET,SO_BROADCAST|SO_REUSEADDR,&optval,sizeof(int));
    struct sockaddr_in theirAddr;
    memset(&theirAddr,0,sizeof(struct sockaddr_in));
    theirAddr.sin_family =AF_INET;
    theirAddr.sin_addr.s_addr =inet_addr("192.168.16.36");//设定接收的地址
    theirAddr.sin_port = htons(8887);//设定接收的端口号
    int sendBytes;
     while(1){
        if((sendBytes=sendto(brdcFd,p_test,sizeof(test_stru),0,(struct sockaddr *)&theirAddr,sizeof(struct sockaddr)))==-1){//sendto将结构体发送出去
            printf("sendto fail,errno=%d\n",errno);
            return -1;
        }
        printf("send success!!!\n");
        //printf("msg=%s,msgLen=%d,sendBytes=%d\n",msg,(int)strlen(msg),sendBytes);
        sleep(5);
    }
    close(brdcFd);  
}

Java server端

import java.io.ByteArrayInputStream;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
class Dgram {

    public static DatagramPacket toDatagram(String s, InetAddress destIA,
            int destPort) {
        byte[] buf = new byte[s.length() + 1];
        s.getBytes(0, s.length(), buf, 0);
        return new DatagramPacket(buf, buf.length, destIA, destPort);
    }

    public static String toString(DatagramPacket p) {
        return new String(p.getData(), 0, p.getLength());
    }
}

 class ChatterServer {
    static final int INPORT = 8887;
    private byte[] buf = new byte[1000];
    private DatagramPacket dp = new DatagramPacket(buf, buf.length);
    private DatagramSocket socket;

    public ChatterServer() {
        try {
            socket = new DatagramSocket(INPORT);//
            System.out.println("Server started");
            while (true) {
                socket.receive(dp);

                String rcvd = Dgram.toString(dp) + ",from address:"
                        + dp.getAddress() + ",port:" + dp.getPort();
                System.out.println("From Client:"+rcvd);
                //dp.getData()获取消息内容
                MAC_struct mac_data = new MAC_struct(dp.getData());
                mac_data.parseFrame();
                String echoString = "From Server Echoed:" + rcvd;
                DatagramPacket echo = Dgram.toDatagram(echoString,
                        dp.getAddress(), dp.getPort());

                socket.send(echo);
            }
        } catch (SocketException e) {
            System.err.println("Can't open socket");
            System.exit(1);
        } catch (IOException e) {
            System.err.println("Communication error");
            e.printStackTrace();
        }
    }
 }
class MAC_struct{
    byte[] recv = new byte[2048];
    public int nodeID;
    public int nodeID2;
    public int stampFlags;
    public long tmp;
    public MAC_struct(byte[] recv){
        this.recv = recv;
    }
    /**
    解析收到的结构体消息。
    */
    public void parseFrame(){
        InputStream in_withcode;
        try {
            for(int i=0;i<24;i++)
                System.out.println("source["+i+"]="+recv[i]);
            in_withcode = new ByteArrayInputStream(recv);
            DataInputStream inputStream  =  new DataInputStream(in_withcode);
            int read=0;
            int updateFlg=0;
            int nodeID=0;
            int[] Bits_4 = new int[4];
            read = inputStream.readUnsignedByte();//读取一个无符号字节
            System.out.println("read= "+read);
            inputStream.skipBytes(3);//跳过N个字节
            updateFlg = inputStream.readInt();//readInt()读取一个4个字节。
            System.out.println("updateFlg= "+int_EndianBigtoLittle(updateFlg));
            nodeID = inputStream.readInt();
            System.out.println("nodeID= "+int_EndianBigtoLittle(nodeID));
            int slotBitMap = inputStream.readUnsignedShort();//读取2个无符号字节。
            System.out.println("slotBitMap= "+uShort_EndianBigtoLittle(slotBitMap));
            inputStream.skipBytes(2);
            System.out.println("Long_num= "+Long_EndianBigtoLittle(inputStream.readInt()));
            for(int i=0;i<4;i++)
                Bits_4[i] = inputStream.readUnsignedByte();
            System.out.println("Float_num= "+Float_EndianBigtoLittle(Bits_4));
        } catch (UnsupportedEncodingException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
    /**
     * 将byte[]数组转为String
     * @param valArr 待转换byte数组,
     * @param startpoint 待转换byte数组开始位置
     * @param maxLen 待转换byte数组长度
     * @return
     */
    private static String toStr(byte[] valArr,int startpoint,int maxLen) {
        int index = 0;
        while(index + startpoint < valArr.length && index < maxLen) {
            if(valArr[index+startpoint] == 0) {
                break;
            }
            index++;
        }
        byte[] temp = new byte[index];
        System.arraycopy(valArr, startpoint, temp, 0, index);
        return new String(temp);
    }
    /**
     * 读取4位,并转化为无符号的4位数据
     * @param value
     * @return 将读取的数据转为小端模式下存储的数据
     */
    private static long Long_EndianBigtoLittle(int value)
    {
        byte[] src = new byte[4];
        src[0] = (byte) ((value>>24)&0xFF);
        src[1] = (byte) ((value>>16)&0xFF);
        src[2] = (byte) ((value>>8)&0xFF);
        src[3] = (byte) ((value)&0xFF);
        return (long) ((src[0]&0xFF)|((src[1]&0xFF)<<8)|((src[2]&0xFF)<<16)|((src[3]&0xFF)<<24))&0xFFFFFFFFl;
    }
    /**
     * 大端数据转小端数据
     * @param value
     * @return 将读取的数据转为小端模式下的数据,
     */
    private static float Float_EndianBigtoLittle(int[] value)
    {
        if (value == null || value.length != 4) {
            throw new IllegalArgumentException("value数组必须不为空,并且是4位!");
        }
        int i = ((value[0]&0xFF)|((value[1]&0xFF)<<8)|((value[2]&0xFF)<<16)|((value[3]&0xFF)<<24));
        return Float.intBitsToFloat(i);
    }
    /**
     * 大端数据转小端数据
     * @param value
     * @return 将读取的数据转为小端模式下的数据,
     */
    private static int int_EndianBigtoLittle(int value)
    {
        byte[] src = new byte[4];
        src[0] = (byte) ((value>>24)&0xFF);
        src[1] = (byte) ((value>>16)&0xFF);
        src[2] = (byte) ((value>>8)&0xFF);
        src[3] = (byte) ((value)&0xFF);
        return (int) ((src[0]&0xFF)|((src[1]&0xFF)<<8)|((src[2]&0xFF)<<16)|((src[3]&0xFF)<<24));
    }
    /**
     * 读物2位数据,转为无符号数据
     * @param value
     * @return 将读取的数据转为小端模式下的数据,
     */
    private static int uShort_EndianBigtoLittle(int value)
    {
        byte[] src = new byte[2];
        src[0] = (byte) ((value>>8)&0xFF);
        src[1] = (byte) ((value)&0xFF);
        return (int) ((src[0]&0xFF)|((src[1]&0xFF)<<8));
    }
    /**
     * 
     */
    public static int vtolh(byte[] bArr) {
        int n = 0;
        for(int i=0;i<bArr.length&&i<4;i++){
            int left = i*8;
            n+= (bArr[i] << left);
        }
        return n;
    }
}
public class JavaCstruct {
    public static void main(String[] args){
        new ChatterServer();
    }
}

上图:
这里写图片描述
如图就是解析C语言传递的结构体数据。
接下来说一下过程,Java server中dp.getData()获取C语言传来的消息内容。将接收到的消息包装成DataInputStream, 接下来就是调用类里面的readXXXX(),获取对应的变量了。这里需要注意的是,要能分析出C语言中,结构体里面的变量在内存里面的存储形式,明白结构体的对齐方式,只有清楚了C的结构体中变量位置,才能正确解析出来里面的数据.
比如在C语言的client程序中,uint8_t是占一个字节,而第二个变量类型是int占四个字节,因为有字节对齐的存在,这两个变量加一起是占了8字节的空间,而不是紧密排列占了5字节空间,uint8_t虽然占了1字节,但是后面的3字节是空着的没有使用,仍然占着空间。假设uint8_t起始地址是0x0000,则结构体中第一个int变量 nodeID占的起始地址是0x0004.所以我们在Java server端中,读取到第1个无符号变量(inputStream.readUnsignedByte();)后,要跳过3个字节(inputStream.skipBytes(3);),再继续读才是下一个变量的开始位置。
还有一个就是大小端转换的问题了,如果client端与server端两者之间存储格式不一样,一端是大端模式,一端是小端模式,那读上来的肯定会不同,我在测验时就是client是小端模式,server是大端模式。如果用我的Java代码读取上来的数据与我的结果不一样,有可能就是这个问题。
当然我们可以从server打印的source[]数组看出端倪。

  • 3
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
Java 通过 UDP 接收 C 语言结构体数组的数据的过程可以分成两个部分,分别是在 C 程序中发送数据和在 Java 程序中接收数据。 在 C 程序中,需要将结构体数组转换成字节流,并通过 UDP 协议发送出去。具体步骤如下: 1. 定义结构体类型,并创建结构体数组: ```c typedef struct { int id; char name[20]; float score; } Student; Student students[3] = {{1, "Tom", 90.5}, {2, "Jerry", 80.5}, {3, "Lucy", 85.0}}; ``` 2. 将结构体数组转换成字节流: ```c char buffer[sizeof(Student) * 3]; memcpy(buffer, students, sizeof(Student) * 3); ``` 3. 创建 UDP 套接字,将字节流发送出去: ```c int sockfd; struct sockaddr_in serveraddr; sockfd = socket(AF_INET, SOCK_DGRAM, 0); bzero(&serveraddr, sizeof(serveraddr)); serveraddr.sin_family = AF_INET; serveraddr.sin_port = htons(PORT); serveraddr.sin_addr.s_addr = inet_addr(IP); sendto(sockfd, buffer, sizeof(buffer), 0, (struct sockaddr *)&serveraddr, sizeof(serveraddr)); ``` 在 Java 程序中,需要接收 UDP 数据包并将字节流转换成结构体数组。具体步骤如下: 1. 创建 DatagramSocket 对象,指定接收端口号: ```java DatagramSocket socket = new DatagramSocket(PORT); ``` 2. 创建 DatagramPacket 对象,用于接收数据包: ```java byte[] buffer = new byte[sizeof(Student) * 3]; DatagramPacket packet = new DatagramPacket(buffer, buffer.length); ``` 3. 接收数据包: ```java socket.receive(packet); ``` 4. 将字节流转换成结构体数组: ```java Student[] students = new Student[3]; for (int i = 0; i < 3; i++) { byte[] data = Arrays.copyOfRange(buffer, i * sizeof(Student), (i + 1) * sizeof(Student)); ByteBuffer byteBuffer = ByteBuffer.wrap(data); students[i] = new Student(byteBuffer.getInt(), new String(byteBuffer.array(), 4, 20), byteBuffer.getFloat()); } ``` 其中,ByteBuffer 类可以方便地将字节流转换成各种数据类型。由于 C 语言中的 char 类型是占用 1 个字节的,而 Java 中的 char 类型是占用 2 个字节的,因此需要将字节流中的 char 类型转换成 String 类型。在这里,我们使用了 String 的构造函数,指定了从字节数组第 4 个位置开始读取 20 个字节,以获取 C 语言中的字符串。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值