0

Java 和 C 之间的套接字有问题。我有一个 C 程序,它创建一个套接字并开始侦听。Java客户端完成它的工作,在某些情况下打开一个套接字并写入它并尝试从中读取,但这会阻塞。

下面是 C 程序的套接字部分:

int sockfd, newsockfd, portno;
socklen_t clilen;
char buffer[256];
char wbuffer[256];
struct sockaddr_in serv_addr, cli_addr;
int n;

sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
        cout << "ERROR opening socket" << endl;
bzero((char *) &serv_addr, sizeof(serv_addr));
portno = 23456;
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(portno);
int on = 1;
if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)
        cout << "Couldn't set socket params!" << endl;
int so = bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr));
while(so < 0) {
        cout << "ERROR on binding" << endl;
        sleep(1);
        so = bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr));
}
listen(sockfd,5);
clilen = sizeof(cli_addr);
newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);
if (newsockfd < 0)
        cout << "ERROR on accept" << endl;
bzero(buffer,256);
//cout << "Socket created!" << endl;
n = read(newsockfd,buffer,255);
if (n < 0)
        cout << "Error reading from socket" << endl;
 if (strcmp(buffer, "power\n") == 0) {
         // Do stuff
     strcpy(wbuffer, "off\n");
         }
     //sleep(1);
     n = write(newsockfd,wbuffer,strlen(wbuffer));
     if (n < 0)
         cout << "ERROR writing to socket" << endl;
}
//cout << "message: " << buffer << endl;
close(newsockfd);
close(sockfd);

Java方面:

Socket socket3 = new Socket("localhost", 23456);
PrintWriter out3 = new PrintWriter(socket3.getOutputStream(), true);
BufferedReader in = new BufferedReader(new InputStreamReader(socket3.getInputStream()));
out3.write("power" + "\n");
//String status = in.readLine();
log.debug("Wrote power");
StringBuffer buffer = new StringBuffer();
while (true) {
    int ch = in.read();
    log.debug("in.read()");
    if ((ch < 0) || (ch == '\n')) {
        break;
    }
    buffer.append((char) ch);
}
String status = buffer.toString();
out3.close();
in.close();
socket3.close();

写作工作,我做了一些调试,它是阅读,特别是 in.read(),这是最后完成的事情。

我可能在 Java 方面遗漏了一些东西,因为 C 测试客户端可以很好地读取输出。

希望有人对如何使这项工作有一些想法,我对任何解决方案持开放态度。

4

3 回答 3

2

您的 C++ 代码非常糟糕。read(2)考虑如果返回小于 then会发生什么6。总的来说:

n = read(newsockfd,buffer,255);

if (n < 0)
    cout << "Error reading from socket" << endl;

// continuing processing after an error - BAD

// not paying attention to n - BAD
if (strcmp(buffer, "power\n") == 0) {
     // Do stuff
     strcpy(wbuffer, "off\n");
}

// potentially writing zero bytes here - BAD
n = write(newsockfd,wbuffer,strlen(wbuffer));

if (n < 0)
     cout << "ERROR writing to socket" << endl;

} // WHERE DID THIS PAREN COME FROM?

所以解决这些问题。检查tcpdump(1)wireshark你在电线上得到的东西。在两面打印您的输入。你可能会发现你的假设在某个地方是错误的。

于 2013-01-15T14:03:14.760 回答
1

事实证明,问题实际上不在阅读部分,而是在写作部分。我忘记刷新输出,所以它实际上并没有开始阅读,或者有什么可以阅读的。添加刷新后读取工作正常,我什至可以使用 readline() 使代码更简单。

我只写入套接字的其他情况有效,因为在关闭流时调用了刷新,这让我失望。

C 代码很讨厌,我需要清理它并实际处理异常。

无论如何,感谢大家帮助我,如果没有答案,发现我的错误需要更长的时间。

于 2013-01-15T15:01:48.643 回答
0

也许问题是你正在阅读一个整数:

int ch = in.read()

您是否尝试将其更改为:

char ch = in.read()

或者在比较之前将整数转换为 char:

if((char) ch == '\n' || ch < 0)

我现在想不出别的了……

于 2013-01-15T13:01:17.150 回答