这是我的带有fork的套接字服务器:
#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);
}
没有分叉服务器和客户端完美地工作但是在我介绍了叉子后我得到了一个奇怪的行为。
让我们试着解释一下:
行为没有分叉:客户端发送消息“Ciao sono il client”,服务器给出响应“Dati ricevuti correttamente”。
行为 with fork:在我按下客户端上的“CTRL-C”之前没有任何事情发生(我在Ubuntu上)。在我按下CTRL-C后,服务器从客户端打印消息
我不明白为什么消息共享不是“按需”,而只是当我停止客户端或服务器时:(
答案 0 :(得分:1)
您的do_child()
代码read
首先来自套接字(然后再recv
之后),在写入之前,而您的父代码根本不包含套接字写代码(至少您提供的代码段没有)因此,当您读取未发送的数据时,do_child()
函数可能会阻塞???