java与c/c++进行socket通信的一些问题

 文章来源: http://blog.csdn.net/kingfish/archive/2005/03/29/333635.aspx

近几天看到csdn上问c/c++和java通信的问题比较多,特别是c特有的数据结构(如struct)。

特地根据网友的一个问题举个例子,希望对初学者有所帮助。

原问题见:http://community.csdn.net/Expert/topic/3886/3886989.xml?temp=.3527033

这类问题通常是为了利用原有Server或者Server不能做修改(通常是c/c++)造成。

比如Server端只接收一个结构Employee,定义如下:

struct UserInfo {
  char UserName[20];
  int UserId;
};
struct Employee {
  UserInfo user;
  float salary;
};
当然也可以定义为

struct Employee {
  char name[20];
  int    id;
  float salary;
};

java client 测试源码(为说明问题,假设struct字节对齐,sizeof(Employee)=28)

import java.net.*;

/**
 * 与C语言通信(java做Client,c/c++做Server,传送一个结构)
 * @author kingfish
 * @version 1.0
 */
class Employee {
  private byte[] buf = new byte[28];  //为说明问题,定死大小,事件中可以灵活处理

  /**
   * 将int转为低字节在前,高字节在后的byte数组
   */
  private static byte[] toLH(int n) {
    byte[] b = new byte[4];
    b[0] = (byte) (n & 0xff);
    b[1] = (byte) (n >> 8 & 0xff);
    b[2] = (byte) (n >> 16 & 0xff);
    b[3] = (byte) (n >> 24 & 0xff);
    return b;
  }

  /**
   * 将float转为低字节在前,高字节在后的byte数组
   */
  private static byte[] toLH(float f) {
    return toLH(Float.floatToRawIntBits(f));
  }

  /**
   * 构造并转换
   */
  public Employee(String name, int id, float salary) {
    byte[] temp = name.getBytes();
    System.arraycopy(temp, 0, buf, 0, temp.length);

    temp = toLH(id);
    System.arraycopy(temp, 0, buf, 20, temp.length);

    temp = toLH(salary);
    System.arraycopy(temp, 0, buf, 24, temp.length);
  }

  /**
   * 返回要发送的数组
   */
  public byte[] getBuf() {
    return buf;
  }

  /**
   * 发送测试
   */
  public static void main(String[] args) {
    try {
      Socket sock = new Socket("127.0.0.1", 8888);
      sock.getOutputStream().write(new Employee("kingfish", 123456789, 8888.99f).
                                   getBuf());
      sock.close();
    }
    catch (Exception e) {
      e.printStackTrace();
    }

} //end

---------------------------------------------------------------------------

当然,也可以利用writeInt,writeFloat方法发送,但字节顺序需要改为低在前。
这个问题稍后在讨论。


如有任何问题,请指正!

kingfish
2005.3.29

-------------------------------------------------------------------------------------------------------

补充说明:

在服务端可更改的情况下就没有这样的麻烦了,现在我也正在写这么一个JAVA客户端、C为服务端的程序,就不需要将字节反序,直接将组好的报文,以字节的方式发送到服务端,然后再取发回来的字节并还原来字符串,到指定位置取相关的信息即可。

我这里涉及到要发送多种不同的报文到服务端,不可能为每个报文发送写一个发送程序,于是就写了一个通用的,用于发送和接收报文,至于报文的组成和接收分析,就采用单独的分析程序,这样可以分开,也便于调试。如图:

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在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、付费专栏及课程。

余额充值