1

我正在尝试让 C99 客户端与 Java 服务器进行通信。但是,Java 服务器接收的数据与传输的数据不同。(即@x�@Óf/ú.��@¨������������������Û¢ÁBp'ˇ¡`Ôf/)

我已经考虑过这是一个编码问题,但是我遇到了障碍。我已经尝试测试这两个程序,以得出 Java 服务器能够与 Java 客户端通信而 C 客户端能够与 C 服务器通信的结论。

但是我无法让 Java 服务器与 C 客户端进行通信。

Java代码:

serverSocket = new ServerSocket(port);  
Socket sock = serverSocket.accept();                               

BufferedReader in = new BufferedReader(new InputStreamReader(sock.getInputStream()));

String inputString = in.readLine();
System.out.println(inputString);

C代码:

struct sockaddr_in sin;
struct hostent *host;

host = gethostbyname(hostname);

bzero(&(sin.sin_zero),8); 
sin.sin_port = htons(port);
sin.sin_addr = *((struct in_addr *)host->h_addr);
sin.sin_family = AF_INET;

sock = socket(AF_INET, SOCK_STREAM, 0);

if(connect(sock, (struct sockaddr *)&sin,sizeof(struct sockaddr_in)) == -1)

...

send(sock, &message, strlen(message)+1, 0);

编辑:我尝试在两台主机之间发送“TEST”这个词,但没有成功。

修复:当我传递消息变量时,我在消息变量前面有一个 & 号。

本来应该:

send(sock, message, strlen(message)+1, 0);
4

4 回答 4

2

我不确定 C99 是否提供了编码标准,但我想它是特定于实现的。可以是US-ASCII例如。

发生的情况是 Java InputStreamReader(或输出)使用在操作系统或 Java 安装(不确定在哪里)中指定的默认字符集,因此与 C 程序使用的字符集相比可能会有所不同。

您可以做的最好的事情是测试各种编码,InputStreamReader该类确实提供了一个特定的构造函数:

public InputStreamReader(InputStream in, Charset cs)

您可以在其中指定要使用的字符集。在这里查看可能的字符集。

于 2012-05-24T04:13:31.317 回答
1

您还必须查看字节序差异:Java 使用网络顺序,而 C 主机可能不使用。ByteBuffer在这里说明,可能会有所帮助。

于 2012-05-24T04:15:49.460 回答
1

如果我没记错的话,你有类似的东西:

char *message = "Hello";

当您通过以下方式发送时:

send(sock, &message, strlen(message)+1, 0);
           |
           +---- Err.

您发送消息的地址而不是消息本身,进一步将长度设置为消息的长度以及这样的垃圾填充。

如果您改为这样做:

send(sock, message, strlen(message)+1, 0);

它应该工作。

或者使用char message[] = "Hello";……</p>


要进一步剖析服务器数据,您可以执行以下操作:

        int i = 0;
        for (char ch : inputString.toCharArray()) {
            System.out.printf("%2d: 0x%02X o%03o %3d %c%n",
                i++,
                (int)ch & 0xff,
                (int)ch & 0xff,
                (int)ch & 0xff,
                Character.isISOControl(ch) ? '.' : ch
            );
        }

或者

        buf = inputLine.getBytes(/* charset  */);
        for (i = 0; i < buf.length; ++i) {

等等

Client siad: Hello
 0: 0x00 o000   0 .
 1: 0x48 o110  72 H
 2: 0x65 o145 101 e
 3: 0x6C o154 108 l
 4: 0x6C o154 108 l
 5: 0x6F o157 111 o

使用 &message:

Client said: `�
 0: 0x60 o140  96 `
 1: 0xEF o357 239 ￯
 2: 0xBF o277 191 ﾿
 3: 0xBD o275 189 ᄑ
 4: 0x04 o004   4 .
 5: 0x08 o010   8 .

另一个可以做的黑客/调试/事情是:

    while (true) {
        in.mark(128);
        if ((inputLine = in.readLine()) == null)
            continue;

        System.out.println("Response: " + inputLine);

        in.reset();
        i = 0;
        while (in.ready()) {
            c = in.read();
            System.out.printf("%2d: 0x%02X o%03o %3d %c%n",
                i++,
                c, c, c,
                Character.isISOControl((char) c) ? '.' : (char)c
            );
        }
    }
于 2012-05-24T06:23:09.727 回答
0

你的问题看起来像字节对齐

在您的C客户端中,您需要将主机转换为网络字节顺序

采用

uint32_t htonl(uint32_t hostlong);

uint16_t htons(uint16_t hostshort);

uint32_t ntohl(uint32_t netlong);

uint16_t ntohs(uint16_t netshort);
于 2012-05-24T04:17:53.793 回答