2012-06-13 45 views
1

這是我的socket服務器叉:交流電插座服務器響應

#include <stdio.h> 
#include <string.h> 
#include <stdlib.h> 
#include <unistd.h> 
#include <sys/socket.h> 
#include <netinet/in.h> 
#include <arpa/inet.h> 
#include <sys/types.h> 
#include <netdb.h> 

void do_child(int sock); 

int main(int argc, char *argv[]){ 
    if(argc != 2){ 
     perror("./server <numero porta>\n"); 
     exit(1); 
    } 
    pid_t pid; 
    int DescrittoreServer, DescrittoreClient, LunghezzaClient; 
    int NumPorta = atoi(argv[1]); 
    struct sockaddr_in serv_addr, cli_addr; /* indirizzo del server e del client */ 
    char Buffer[1024] = {}; 
    DescrittoreServer = socket(AF_INET, SOCK_STREAM, 0); 
    if(DescrittoreServer < 0){ 
     perror("Errore: creazione socket\n"); 
     exit(1); 
    } 
    bzero((char *) &serv_addr, sizeof(serv_addr)); /* bzero scrive dei null bytes dove specificato per la lunghezza specificata */ 
    serv_addr.sin_family = AF_INET; /* la famiglia dei protocolli */ 
    serv_addr.sin_port = htons(NumPorta); /* porta */ 
    serv_addr.sin_addr.s_addr = INADDR_ANY; /* dato che è un server bisogna associargli l'indirizzo della macchina su cui sta girando */ 

    if(bind(DescrittoreServer, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0){ 
     perror("Errore di bind\n"); 
     close(DescrittoreServer); 
     exit(1); 
    } 
    listen(DescrittoreServer, 5); 
    LunghezzaClient = sizeof(cli_addr); 
    while(1){ 
     DescrittoreClient = accept(DescrittoreServer, (struct sockaddr *) &cli_addr, &LunghezzaClient); 
     if(DescrittoreClient < 0){ 
      perror("Errore: non è possibile stabilire la connessione\n"); 
      close(DescrittoreServer); 
      close(DescrittoreClient); 
      exit(1); 
     } 
     pid = fork(); 
     if(pid < 0){ 
      perror("Fork error"); 
      close(DescrittoreServer); 
      close(DescrittoreClient); 
      exit(1); 
     } 
     if(pid == 0){ 
      close(DescrittoreServer); 
      do_child(DescrittoreClient); 
      exit(0); 
     } 
     else{ 
      close(DescrittoreClient); 
     } 
    } 
} 

void do_child(int sock){ 
    int n; 
    char Buffer[1024] = {}; 
    n=read(sock, Buffer, 255); 
    if(n < 0){ 
     perror("Errore nella lettura dalla socket\n"); 
     exit(1); 
    } 
    recv(sock, Buffer, sizeof(Buffer), 0); 
    printf("Dati ricevuti: %s\n", Buffer); 
    strcpy(Buffer, "Dati ricevuti correttamente!"); 
    send(sock, Buffer, strlen(Buffer)+1, 0); 
} 

沒有叉的服務器和客戶端的工作完美,但我已經介紹了叉後,我從他們那裏得到了一個奇怪的現象。
讓我們試着解釋一下:
行爲沒有 fork:客戶端發送消息「Ciao sono il client」,服務器給出響應「Dati ricevuti correttamente」。
行爲 fork:直到我按下客戶端上的「CTRL-C」(我在Ubuntu上)才發生任何事情。在我按下CTRL-C後,服務器從客戶端打印消息。
我不明白爲什麼共享這些信息並不「需求」,但只有當我停止客戶端或服務器:(

回答

1

do_child()代碼從插座read第(再經過這個recv再次) ,而你的父母代碼根本不包含任何套接字寫入代碼(至少你提供的代碼段沒有),所以當你讀取數據時,你的do_child()函數可能被阻塞了嗎?

+0

你是對的!我已經刪除了「閱讀」,現在都正常工作:D謝謝「 – polslinux