【问题标题】:EPIPE error does not appear?EPIPE错误不出现?
【发布时间】:2014-02-03 21:01:26
【问题描述】:

根据我阅读的有关 SIGPIPE 的内容,我进行了测试以尝试产生 SIGPIPE 问题。下面是服务端和客户端的代码:

服务器代码:

#include <stdio.h>
#include <stdlib.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <errno.h>
#include <unistd.h>
#include <fcntl.h>

void error(char *msg)
{
    perror(msg);
    exit(1);
}

int main(int argc, char *argv[])
{
     int sockfd, newsockfd, portno, clilen;
     char buffer[256];
     struct sockaddr_in serv_addr, cli_addr;
     int n;

     if (argc < 2) {
         fprintf(stderr,"ERROR, no port provided\n");
         exit(1);
     }

     sockfd = socket(AF_INET, SOCK_STREAM, 0);

     if (sockfd < 0)
        error("ERROR opening socket");

     bzero((char *) &serv_addr, sizeof(serv_addr));
     portno = atoi(argv[1]);
     serv_addr.sin_family = AF_INET;
     serv_addr.sin_addr.s_addr = INADDR_ANY;
     serv_addr.sin_port = htons(portno);

     if (bind(sockfd, (struct sockaddr *) &serv_addr,
              sizeof(serv_addr)) < 0)
              error("ERROR on binding");

     listen(sockfd,5);

     clilen = sizeof(cli_addr);
     newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);

     if (newsockfd < 0)
          error("ERROR on accept");
     bzero(buffer,256);

     n = read(newsockfd,buffer,255);

     if (n < 0)
         error("ERROR reading from socket");

     printf("Here is the message: %s\n",buffer);

     n = write(newsockfd,"I got your message",18);

     if (n < 0)
         error("ERROR writing to socket");
     else
         error("no error");
         return 0;
}

客户端代码:

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <errno.h>
#include <signal.h>
#include <resolv.h>
#include <arpa/inet.h>
#include <unistd.h>

void error(char *msg)
{
    perror(msg);
    exit(0);
}

int main(int argc, char *argv[])
{
    int sockfd, portno, n;

    struct sockaddr_in serv_addr;
    struct hostent *server;

    char buffer[256];
    if (argc < 3) {
       fprintf(stderr,"usage %s hostname port\n", argv[0]);
       exit(0);
    }
    portno = atoi(argv[2]);
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
        error("ERROR opening socket");
    server = gethostbyname(argv[1]);
    if (server == NULL) {
        fprintf(stderr,"ERROR, no such host\n");
        exit(0);
    }
    bzero((char *) &serv_addr, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    bcopy((char *)server->h_addr,
         (char *)&serv_addr.sin_addr.s_addr,
         server->h_length);
    serv_addr.sin_port = htons(portno);
    if (connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr)) < 0)
        error("ERROR connecting");
    printf("Please enter the message: ");
    bzero(buffer,256);
    fgets(buffer,255,stdin);

    signal(SIGPIPE, SIG_IGN);

    n = write(sockfd,buffer,strlen(buffer));

    printf("n=%d, errno=%d", n, errno);

    bzero(buffer,256);
    n = read(sockfd,buffer,255);

    printf("n=%d, errno=%d", n, errno);

    printf("%s\n",buffer);
    return 0;
}

我设置实验的方式如下。我先启动服务器,然后启动客户端。在客户端打印出消息:“请输入消息:”后,我停止了服务器。然后我尝试在客户端输入一些文本(例如“test”)进行发送。因为我使用了signal(SIGPIPE, SIG_IGN),所以我期望的是来自客户端的一些EPIPE 错误代码,但我什么也没得到。客户端打印出来:

Please enter the message: test
n=5, errno=0
n=0, errno=0

我错过了什么吗?如何设置写操作的代码返回EPIPE错误?

【问题讨论】:

    标签: c sockets tcp sigpipe epipe


    【解决方案1】:

    不要尝试读取,只需写入更多数据。发送失败可能需要一段时间。

    【讨论】:

    • 为什么它不能在第一个write()返回错误?在实际应用中,如果我只有一个write(),如何捕获EPIPE错误?
    • 因为还不知道对方已经关闭了
    • 在我的应用程序中,我希望服务器捕获错误以做一些工作(清理客户端请求产生的一些数据)。我怎么能在第一次write() 通话中做到这一点?
    • 我不确定您指的是哪种错误。在您的示例中,您希望客户端写入失败,因为服务器关闭 - 那么如果您明确关闭它,服务器为什么要清理任何东西?无论如何:如果本地写入失败,写入将返回错误,例如内核无法处理它,因为它在用户空间发送数据之前从另一端关闭。它不会伸手到另一边,检查那里是否收到数据然后返回。
    猜你喜欢
    • 1970-01-01
    • 2014-12-26
    • 2021-03-04
    • 1970-01-01
    • 1970-01-01
    • 2016-04-25
    • 2017-08-10
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多