我正在学习 c 上的套接字。我有一个客户端和一个服务器,当服务器关闭套接字时,客户端仍然能够在发送获得 SIGPIPE 信号之前接收并向服务器发送另外两个包。我不知道为什么。有谁可以帮忙吗~
因为文档说如果send
和recv
有错误,那么它们将返回-1。但在我这里,这种情况从未发生过。
客户端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define BUFFERSIZE 255
#define MAXLENG 96
#define true 1
#define false 0
void exitp();
int main(int argc, char *argv[]){
signal(SIGPIPE, exitp);
// Check if the argument is match the requirement
if (argc != 3) {
fprintf(stderr, "Useage Error, should be follow by ip, and port\n" );
exit(1);
}
// Create the socket for the client, if the fd for the socket == -1, it means
// it created fail
int skfd = socket(AF_INET, SOCK_STREAM, 0);
if (skfd < 0){
fprintf(stderr, "Create socket failed\n" );
exit(1);
}
int PORT = atoi(argv[2]);
// Set up server argument
struct sockaddr_in server_addr;
memset(&server_addr, '0', sizeof(server_addr)); // addr for bin
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(PORT);
// ip not vaild
if (inet_pton(AF_INET, argv[1], &server_addr.sin_addr) < 1){
fprintf(stderr, "IP address not correct\n" );
exit(1);
}
// create the connection
if (connect(skfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1){ // Connect to server
fprintf(stderr, "Connection Fail\n" );
exit(1);
}
char sentbuff[255], recvbuffer[255], input[255], request[100], concelbuffer[255];
// Send the request to the server
int nn;
int size = send(skfd, request, sizeof(request),0);
while(1){
if (recv(skfd, &recvbuffer,sizeof(recvbuffer),0) == -1){
printf("Server Closed\n");
break;
}
printf("%s",recvbuffer);
fgets(sentbuff, 255, stdin);
nn=send(skfd, &sentbuff, sizeof(sentbuff), 0);
if (nn == -1){
printf("Server Closed\n");
break;
}
}
return 0;
}
void exitp(){
printf("%s\n","Server Closed" );
exit(0);
}
服务器端我使用了shutdown
和close
作为acceptfd
最佳答案
send()
只是将数据放入内核套接字缓冲区,它不会等待数据传输或服务器确认收到数据。
直到数据传输完毕并且服务器通过发送 RST
段拒绝数据后,您才会收到 SIGPIPE
。
它以这种方式工作是因为 TCP 连接的每个方向都是独立处理的。当服务器关闭套接字时,它会发送一个FIN
段。这只是告诉客户端服务器已完成发送数据,并不意味着服务器无法接收数据。 TCP 协议(protocol)中没有任何内容允许服务器将此情况通知客户端。因此,发现它不再接受任何数据的唯一方法是当客户端收到 RST
响应时。
通知客户端不应再发送任何内容通常是在应用程序协议(protocol)中完成的,因为它在 TCP 中不可用。
关于C 服务器套接字已关闭,但客户端套接字仍然可以再发送两个包,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49457631/