C |套接字未正确关闭

时间:2018-07-05 14:29:45

标签: c sockets fork telnet

我当前正在编写一台使用fork处理多个连接的服务器。

我正在通过telnet测试它。一切正常,但是关闭连接时telnet不会告诉我。

这是我的代码:

int main(void) {
    int pop3socket;
    int clientSocket;
    int len;
    int binden;
    int zuhoeren;
    int schliessen;
    int enable=1;
    int pid;
    int pop3_status;
    int server_running = 1;
    /* Infos ueber die Kommunikationspartner :) */
    struct sockaddr_in mailserver;
    struct sockaddr_in pop3client;

    /* Socket erstellen */
    pop3socket = socket(AF_INET, SOCK_STREAM, 0);
    if(pop3socket < 0) {
        perror("Fehler beim Erstellen des Sockets!");
    }

    /* IP wiederverwenden */
    if(setsockopt(pop3socket, SOL_SOCKET, SO_REUSEADDR, &enable, sizeof(int)) < 0) {
        perror("Fehler beim Wiederverwenden der IP");
    }

    /* IP und Port festlegen mithilfe von Bind */
    memset(&mailserver, 0, sizeof(struct sockaddr_in)); 
    mailserver.sin_family = AF_INET;
    mailserver.sin_addr.s_addr = htonl(INADDR_ANY);
    mailserver.sin_port = htons(PORT);
    binden = bind(pop3socket, (struct sockaddr*)&mailserver, sizeof(mailserver));
    if(binden < 0) {

    }

    /* Warteschlange einrichten */
    zuhoeren = listen(pop3socket, 5);
    if(zuhoeren == -1) {
        perror("Fehler beim Listen");
    }

    /* Verbindung(en) akzeptieren */
    for(;;) {
        len = sizeof(struct sockaddr);
        clientSocket = accept(pop3socket, (struct sockaddr*)&pop3client, &len);

        if(clientSocket < 0) {
            perror("Fehler beim Accepten der Verbindung");
        }

        if( (pid = fork()) < 0) {
            perror("Fork fehlgeschlagen");
            exit(8);
        } else if(pid == 0) {
            close(pop3socket);
            printf("Prozess: %d\n", getpid());
            /* Begruessen */
            sendSuccessMessage("", clientSocket);
            while(server_running == 1) {
                switch( (pop3_status = (process_pop3(clientSocket, clientSocket))) ) {
                    printf("Status: %d\n", pop3_status);
                    case -1: 
                        server_running = 0;
                        break;
                }
            }
            if(server_running == 0) {
                printf("Connection beenden\n");
                schliessen = close(clientSocket);
                if(schliessen < 0) {
                    perror("Fehler beim Schließen des Sockets");
                    exit(9);
                }
                exit(0);
            }
        }
    }
    return 0;
}

似乎我的子进程一直在运行,并且无法正常结束? 我完全不知道为什么会这样。

这是telnet输出:

telnet localhost 9999 Trying 127.0.0.1... Connected to localhost. Escape character is '^]'.
+OK  user joendhard
+OK  pass test123
+OK User logged in. stat
+OK 9 9735 quit
+OK User logged out.

预期输出:

telnet本地主机9999 正在尝试127.0.0.1 ... 连接到本地主机。 转义字符为'^]'。 +确定 用户joendhard +确定 通过test123 +确定用户登录。 统计 +确定9 9735 放弃 +确定用户已注销。

telnet localhost 9999
Trying 127.0.0.1...
Connected to localhost.
Escape character is '^]'.
+OK 
user joendhard
+OK 
pass test123
+OK User logged in.
stat
+OK 9 9735
quit
+OK User logged out.
Connection closed by foreign host.

谢谢!

1 个答案:

答案 0 :(得分:3)

客户端套接字在父进程中仍处于打开状态。分叉后在父进程中将其关闭。

当所有引用套接字文件 description 的文件 descriptor 关闭时,套接字终止连接(通过调用shutdown)。您这里有2个文件描述符,它们是指客户端套接字文件 description