私はこの例をCスレッドサーバーとJavaクライアントで処理しています。これはサーバです:Javaクライアントの接続後にCサーバーがクラッシュする
#include <stdio.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <pthread.h>
void* thread_proc(void *arg);
int main(int argc, char *argv[])
{
struct sockaddr_in sAddr;
int listensock;
int result;
int nchildren = 6;
pthread_t thread_id;
int x;
int val;
listensock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
val = 1;
result = setsockopt(listensock, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val));
if (result < 0) {
perror("server");
return 0;
}
sAddr.sin_family = AF_INET;
sAddr.sin_port = htons(6000);
sAddr.sin_addr.s_addr = INADDR_ANY;
result = bind(listensock, (struct sockaddr *) &sAddr, sizeof(sAddr));
if (result < 0) {
perror("exserver5");
return 0;
}
result = listen(listensock, 5);
if (result < 0) {
perror("exserver5");
return 0;
}
for (x = 0; x < nchildren; x++) {
result = pthread_create(&thread_id, NULL, thread_proc, (void *) listensock);
if (result != 0) {
printf("Could not create thread.\n");
return 0;
}
sched_yield();
}
pthread_join (thread_id, NULL);
}
void* thread_proc(void *arg)
{
int listensock, sock;
char buffer[25];
int nread;
listensock = (int) arg;
while (1) {
sock = accept(listensock, NULL, NULL);
printf("client connected to child thread %i with pid %i.\n", pthread_self(), getpid());
while(1){
nread = recv(sock, buffer, 25, 0);
buffer[nread] = '\0';
printf("%s\n", buffer);
send(sock, buffer, nread, 0);
if(nread == '9'){
close(sock);
}
}
printf("client disconnected from child thread %i with pid %i.\n", pthread_self(), getpid());
}
}
これは、Javaクライアントである:
import java.net.*;
import java.io.*;
// A client for our multithreaded EchoServer.
public class client
{
public static void main(String[] args)
{
Socket s = null;
// Create the socket connection to the EchoServer.
try
{
s = new Socket("localhost", 6000);
}
catch(UnknownHostException uhe)
{
// Host unreachable
System.out.println("Unknown Host");
s = null;
}
catch(IOException ioe)
{
// Cannot connect to port on given host
System.out.println("Cant connect to server at 6000. Make sure it is running.");
s = null;
}
if(s == null)
System.exit(-1);
BufferedReader in = null;
PrintWriter out = null;
try
{
// Create the streams to send and receive information
in = new BufferedReader(new InputStreamReader(s.getInputStream()));
out = new PrintWriter(new OutputStreamWriter(s.getOutputStream()));
// Since this is the client, we will initiate the talking.
// Send a string.
out.println("Hello");
out.flush();
// receive the reply.
System.out.println("Server Says : " + in.readLine());
// Send a string.
out.println("This");
out.flush();
// receive a reply.
System.out.println("Server Says : " + in.readLine());
// Send a string.
out.println("is");
out.flush();
// receive a reply.
System.out.println("Server Says : " + in.readLine());
// Send a string.
out.println("a");
out.flush();
// receive a reply.
System.out.println("Server Says : " + in.readLine());
// Send a string.
out.println("Test");
out.flush();
// receive a reply.
System.out.println("Server Says : " + in.readLine());
// Send the special string to tell server to quit.
out.println("9");
out.flush();
}
catch(IOException ioe)
{
System.out.println("Exception during communication. Server probably closed connection.");
}
finally
{
try
{
// Close the streams
out.close();
in.close();
// Close the socket before quitting
s.close();
}
catch(Exception e)
{
e.printStackTrace();
}
}
}
}
私は、クライアントを実行すると、これが出力されます:
[[email protected] java]# /opt/jdk1.7.0_03/bin/java client
Server Says : Hello
Server Says : This
Server Says : is
Server Says : a
Server Says : Test
[[email protected] java]#
私は、サーバーを実行すると、これはあります出力:
[[email protected] java]# ./server
client connected to child thread -1215423632 with pid 2854.
Hello
This
is
a
Test
9
9
[[email protected] java]#
ientがサーバーがクラッシュする接続を閉じます。私はスレッドを破壊することなく接続を閉じたい。どのように問題を解決できますか?
ベストこれは接続を受け入れ、元のスレッドコードである
P.Sを望みます。
void* thread_proc(void *arg)
{
int listensock, sock;
char buffer[25];
int nread;
listensock = (int) arg;
while (1) {
sock = accept(listensock, NULL, NULL);
printf("client connected to child thread %i with pid %i.\n", pthread_self(), getpid());
nread = recv(sock, buffer, 25, 0);
buffer[nread] = '\0';
printf("%s\n", buffer);
send(sock, buffer, nread, 0);
close(sock);
printf("client disconnected from child thread %i with pid %i.\n", pthread_self(), getpid());
}
}
このループは1つの文字列のみを受け入れ、それを返します。その後、終了します。私はwhileループでそれを修正しますが、内側のwhileループ(上記参照)ではサーバーがクラッシュします。 スレッドが多くの文字列を受け入れ、多くの文字列を送信できるように、コードを変更する方法はありますか?
とのrecvをのためにチェックする必要があります0(相手側が接続を閉じたことを意味する)または-1(エラー)を返します。チャンスは 'nread == '9' 'も間違っています。受信したバイト数が57であるかどうかをチェックします。 – nos
+1素晴らしいキャッチです。しかし、これはスレッドを終了させ、それはOPが望んでいないものです – GETah
いいえ、それは内部のループを壊すだけです – nos