为什么 CTRL-D 信号没有使用此程序进行 EOF?[复制]

Why isn't CTRL-D signaling EOF with this program? [duplicate]

提问人:user2264035 提问时间:12/3/2013 最后编辑:Jim Lewisuser2264035 更新时间:12/3/2013 访问量:5189

问:

当我运行这个程序时,我输入一个IP地址和服务器端口作为输入。但是,在此之后,程序要求我“请输入msg:”。但是,在键入我的消息后,按 ctrl-D 不会执行任何操作。程序仍从 STDIN 读取。如何指示我正在输入的消息的结尾?

这是我正在使用的程序:

/* 
 * echoclient.c - A simple connection-based client
 * usage: echoclient <host> <port>
 */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h> 

#define BUFSIZE 1024

/* 
 * error - wrapper for perror
 */
void error(char *msg) {
    perror(msg);
    exit(0);
}

int main(int argc, char **argv) {
    int sockfd, portno, n;
    struct sockaddr_in serveraddr;
    struct hostent *server;
    char *hostname;
    char buf[BUFSIZE];

    /* check command line arguments */
    if (argc != 3) {
       fprintf(stderr,"usage: %s <hostname> <port>\n", argv[0]);
       exit(0);
    }
    hostname = argv[1];
    portno = atoi(argv[2]);

    /* socket: create the socket */
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) 
        error("ERROR opening socket");

    /* gethostbyname: get the server's DNS entry */
    server = gethostbyname(hostname);
    if (server == NULL) {
        fprintf(stderr,"ERROR, no such host as %s\n", hostname);
        exit(0);
    }

    /* build the server's Internet address */
    bzero((char *) &serveraddr, sizeof(serveraddr));
    serveraddr.sin_family = AF_INET;
    bcopy((char *)server->h_addr, 
      (char *)&serveraddr.sin_addr.s_addr, server->h_length);
    serveraddr.sin_port = htons(portno);

    /* connect: create a connection with the server */
    if (connect(sockfd, (struct sockaddr *)&serveraddr, sizeof(serveraddr)) < 0) 
      error("ERROR connecting");

    /* get message line from the user */
    printf("Please enter msg: ");
    bzero(buf, BUFSIZE);
    fgets(buf, BUFSIZE, stdin);


    /* write: send the message line to the server */
    n = write(sockfd, buf, strlen(buf));
    if (n < 0) 
      error("ERROR writing to socket");

    /* read: print the server's reply */
    bzero(buf, BUFSIZE);
    n = read(sockfd, buf, BUFSIZE);
    if (n < 0) 
      error("ERROR reading from socket");
    printf("Echo from server: %s", buf);
    close(sockfd);
    return 0;
}
C 套接字 UNIX stdin EOF

评论

1赞 jwodder 12/3/2013
看起来程序应该在您点击消息第一行末尾的“Enter”后立即停止阅读。是什么让你认为它仍在读取输入?stdin
0赞 user2264035 12/3/2013
如果我按回车键,它只会转到下一行并继续接收更多输入。
0赞 Tim Pierce 12/3/2013
我怀疑该程序已经继续并正在等待从服务器获得响应,同时没有收听来自键盘的输入。此时,您可以在键盘上键入任何您想要的内容,包括 Ctrl-D,但在调用返回之前,程序不会注意到它。n = read(sockfd, buf, BUFSIZE)read
0赞 user2264035 12/3/2013
如果我想要来自服务器的响应,我是先于客户端运行服务器程序,还是先于服务器运行客户端程序?
0赞 Tim Pierce 12/3/2013
如果在启动服务器之前运行客户端程序,则客户端将没有任何内容可以连接:-)

答:

0赞 chux - Reinstate Monica 12/3/2013 #1

需要测试结果fgets(buf, BUFSIZE, stdin);

如果不测试结果,可能会包含其未更改的先前数据,并且不会提供不再读取数据的明确信息。buf

评论

0赞 user2264035 12/3/2013
如何测试结果?
0赞 Potatoswatter 12/3/2013
@user2264035 .if ( result = fgets(…) ) …
0赞 chux - Reinstate Monica 12/3/2013
@user2264035if (fgets(buf, BUFSIZE, stdin) == NULL) Handle_EOF_or_IOError();
4赞 Jonathan Leffler 12/3/2013 #2

如果您在行上键入了任何数据,则第一个会将您键入的内容发送到终端,尽管您还没有按回车键。然后,您需要再次键入以发送 0 个字符,这表示 EOF 到标准 I/O 库。Control-DControl-D

如果您键入的最后一个字符是回车符,则保存的数据无论如何都会发送到终端。如果下一个字符是 ,则不发送任何数据,因此读取获取 0 字节,并将其检测为 EOF。Control-D

另请参阅规范与非规范终端输入

您还应该始终检查返回值 from 和任何其他输入函数。如果输入函数报告失败,则无法使用输入;其状态未定义。fgets()

0赞 R.. GitHub STOP HELPING ICE 12/3/2013 #3

fgets读取直到它到达 EOF 或换行符,就像调用每个字节一样。按下终端不会在终端上“设置EOF状态”(没有这样的东西)或“发送和EOF”(也没有这样的东西)。它的作用(在“熟模式”下)是让内核通过 tty 设备发送所有缓冲的行编辑内容,以便应用程序接收它。如果还没有任何输入,这将导致应用程序成功读取长度为零的 ;这是 EOF 的定义,将设置 stdio 流的 EOF 状态。fgetcCtrlDreadFILE

但是,如果已经有输入等待(您输入的行),则只会将其发送到应用程序。 然后将读取所有传输的字节,并且,在看不到 EOF 或换行符的情况下,将像再次调用一样,再次阻塞 .如果再按一次或按 ,则成功并返回。CtrlDfgetsfgetcreadCtrlDEnterfgets