С-сервер падает после подключения клиента Java - PullRequest
0 голосов
/ 04 марта 2012

Я работаю над этим примером с многопоточным сервером C и клиентом Java.Это сервер:

#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();
            }                
        }        
    }
} 

Когда я запускаю клиент, это вывод:

[root@localhost java]# /opt/jdk1.7.0_03/bin/java client
Server Says : Hello
Server Says : This
Server Says : is
Server Says : a
Server Says : Test
[root@localhost java]# 

Когда язапустите сервер, это вывод:

[root@localhost java]# ./server 
client connected to child thread -1215423632 with pid 2854.
Hello

This

is

a

Test

9

9

[root@localhost java]# 

Когда клиент закрывает соединение, сервер падает.Я хочу закрыть соединение, не разрушая поток.Как я могу решить проблему?

С наилучшими пожеланиями

PS Это исходный код темы, который принимает соединения.

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());
  }
}

Этот цикл будет принимать только одну строкуверни это.Тогда выйдет.Я изменяю его с помощью цикла while, но с внутренним циклом while (см. Выше) сервер падает.Как я могу изменить код, чтобы поток мог принимать много строк и отправлять много строк?

Ответы [ 2 ]

3 голосов
/ 04 марта 2012

Вам нужно выйти из внутреннего цикла после того, как вы закроете сокет - вы терпите крах, пытаясь прочитать из закрытого сокета:

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);
            break;       // <--- need to break out here
        }
    }
1 голос
/ 04 марта 2012
while (1) {
    sock = accept(listensock, NULL, NULL);
    printf("client connected to child thread %i with pid %i.\n", pthread_self(), getpid());

    nread = 0;
    while(nread >= 0) {
        nread = recv(sock, buffer, 25, 0);
        if (nread >= 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());

}

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...