2010-10-27 219 views
1

我正在将我的代码粘贴到C和Java客户端中的简单套接字服务器上。C套接字服务器,Java套接字客户端:阻塞!

我使用写入方法在Java中逐字发送。然而,在发送大块字符(在这里,'h','e','y')后,Java客户端被发送阻塞,因为C服务器不回复它:(

我假设有一些问题,发送一个空字符(从Java写),将在C面停止的recv

任何帮助,将不胜感激

C服务器:。

#include <stdio.h> /* standard in and output*/ 
#include <sys/socket.h> /* for socket() and socket functions*/ 
#include <arpa/inet.h> /* for sockaddr_in and inet_ntoa() */ 
#include <stdlib.h>  
#include <string.h>  
#include <unistd.h>  /* for close() */ 


int main(int argc, char *argv[]){ 
int sock, connected, bytes_received, true = 1; 
    char recv_data;  
char replyBuffer[32]; 

    struct sockaddr_in server_addr,client_addr;  
    int sin_size;  
    if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1) { 
     perror("Socket"); 
     exit(1); 
    } 

    if (setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&true,sizeof(int)) == -1) { 
     perror("Setsockopt"); 
     exit(1); 
    } 

    server_addr.sin_family = AF_INET;   
    server_addr.sin_port = htons(2400);  
    server_addr.sin_addr.s_addr = INADDR_ANY; 
    bzero(&(server_addr.sin_zero),8); 

    if (bind(sock, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) { 
     perror("Unable to bind"); 
      exit(1); 
    } 

    if (listen(sock, 5) == -1) { 
     perror("Listen"); 
     exit(1); 
    } 

printf("\nTCPServer Waiting for client on port 2400"); 

    while(1){ 
     sin_size = sizeof(client_addr); 
     connected = accept(sock, (struct sockaddr *)&client_addr,&sin_size); 
     printf("\n Got a connection from (%s , %d)",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port)); 


     while ((bytes_received = recv(connected,&recv_data,1,0)) > 0){ 
    printf("\nrecv= %c\n", recv_data); 
    } 
    int success = 1; 
    sprintf(replyBuffer, "%d", success); 
    printf("reply buffer = %s\n", replyBuffer); 
    if (send(connected, replyBuffer, strlen(replyBuffer), 0) == -1) 
    perror("send() failed"); 
    success = 0; 
    close(connected); 
} 
} 

Java客户端:

import java.net.*; 
import java.io.*; 

public class Client1 
{ 
public static void main(String[] args) throws IOException { 
    if (args.length < 2) { 
    System.err.println("Usage: java Client1 <IP address> <Port number>"); 
    System.exit(0); 
    } 
    BufferedReader in = null; 
    OutputStream out = null; 
    Socket sock = null; 

    try { 
    sock = new Socket(args[0], Integer.parseInt(args[1])); 
    out = sock.getOutputStream(); 
    in = new BufferedReader(new InputStreamReader(sock.getInputStream())); 

    String line = "hey"; 
    String responseline = null; 
char[] strArray; 
strArray = line.toCharArray(); 

while (true) { 
for(int index = 0; index < strArray.length; index++){ 
out.write(strArray[index]); 
} 
out.flush(); 
System.out.println("data sent "); 
System.out.println("val returned"+in.readLine()); 
    } 
    } 
    catch (IOException ioe) { 
    System.err.println(ioe); 
    } 
    finally { 
    if (in != null) 
    in.close(); 

    if (out != null) 
    out.close(); 
    if (sock != null) 
    sock.close(); 
    } 
} 
} 
+0

@ user489152:你应该接受一个答案。 – andersoj 2010-11-02 14:10:57

回答

8

从Javadoc文档BufferedReader.readLine()

Read a line of text. A line is considered to be terminated by any one of a line feed ('\n'), a carriage return ('\r'), or a carriage return followed immediately by a linefeed. 

这就是为什么它块(不LF,没有CR,没有CRLF)

[编辑]
你需要创建一个EOL指示C程序可以理解(就像在java客户端中的readLine()一样)。让该指标为'\ n'(为了一致性,如果没有其他)
在Client.java中,将'\ n'附加到您发送的字符串。
在tst.c,测试 '\ n',它在接收时,突破来自的recv()循环
[/编辑]

+0

但是我试着发送'/ n','/ 0','/ r',这些都没有工作......最后我甚至尝试传递'null',然后服务器响应,但是立即,我的java客户端停止了可以理解的在零点异常之后。 – user489152 2010-10-28 06:16:08

+0

谢谢凯文,我做了这些,现在C代码进一步发展,在它接收到整个字符集之后,它返回一个值(replyBuffer中success = 1),但是这次java代码块中的in.readLine() !但它不承认空字符?我甚至把in.readLine()放在while循环中,直到它不等于null ...但它会阻止...:(:( – user489152 2010-10-28 16:24:56

+0

在两个方向的通信必须'\ n'终止(这是你的EOL指标在两边) - sprintf(replyBuffer,“%d \ n”,成功); – KevinDTimm 2010-10-28 16:36:47

2

你的问题的假设是正确的。调用recv将始终返回大于零的值。我建议使用特殊字符指定文本的结尾,并在读取该字符时终止recv()循环。

+0

但我试着发送'/ n','/ 0','/ r',这些都没有工作...最后我甚至尝试过传递'空',然后服务器响应,但立即,我的Java客户端停止空点异常后可以理解。 – user489152 2010-10-28 06:14:48

+0

史蒂夫,你能提出一些特殊的特征吗?顺便说一句,我会用这个程序来传递ASCII文件(文本)...任何建议? – user489152 2010-10-28 06:15:44

+0

一旦EOL终结符被定义,发送和识别,这一切都很好:) – KevinDTimm 2010-10-28 12:21:50

0

在我看来,你的服务器代码有错误。如果你在linux上编译你的服务器代码,会出现错误。 sin_size的类型应该是socklen_t。 或者你会看到一个错误信息 “从int *无效的转换为socklen_t”

0

客户改变

out.write(line.getBytes()); 

服务器更改为

char recv_data[32]; 
recv(connect,recv_data,32,0);