TCP客戶/服務器程序實例——回射服務器


目錄

客戶/服務器程序源碼

POSIX信號處理

POSIX信號語義

處理SIGCHLD信號

處理僵死進程

處理被中斷的系統調用

wait和waitpid函數

wait和waitpid函數的區別

網絡編程可能會遇到的三種情況

TCP程序小結

數據格式

 

回射輸入行這樣一個客戶/服務器程序是一個雖然簡單然而卻很有效的網絡應用程序的例子。實現任何客戶/服務器網絡應用所需的所有基本步驟可通過本例子闡明。若想把本例子擴充成你自己的應用程序,你只需修改服務器對於來自客戶的輸入的處理過程。

image

TCP回射服務器程序:main函數

/* tcpserv01.c */
#include <sys/socket.h>
#include <strings.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>

int
main(int argc, char **argv)
{
    int                   listenfd, connfd;
    pid_t                 childpid;
    socklen_t             clilen;
    struct sockaddr_in    cliaddr, servaddr;
    
    if((listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
    {
        perror("socket");
        exit(1);
    }
    
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(9877);

    if(bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)
    {
        perror("bind");
        exit(1);
    }
    if(listen(listenfd, 5) < 0)
    {
        perror("listen");
        exit(1);
    }

    for(;;)
    {
        clilen = sizeof(cliaddr);
        if((connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &clilen)) < 0)
        {
            perror("accept");        
            exit(1);
        }
        
        if((childpid = fork()) < 0)
        {
            perror("fork");
            exit(1);
        }
        else if(childpid == 0)    /* child process */
        {
            if(close(listenfd) < 0) /* close listening socket */
            {
                perror("child close");
                exit(1);
            }
            str_echo(connfd);    /* process the request */
            exit(0);
            
        }
        if(close(connfd) < 0) /* parent close connected socket */
        {
            perror("parent close");
            exit(1);
        }
    }
    
    
}

TCP回射服務器程序:str_echo函數

/* str_echo.c */
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>

void
str_echo(int sockfd)
{
    ssize_t        n;
    char           buf[4096];
again:
    while((n = read(sockfd, buf, 4096)) > 0)
        writen(sockfd, buf, n);

    if(n < 0 && errno == EINTR)
        goto again;
    else if(n < 0)
    {
        perror("read");
        exit(1);
    }
        
}

TCP回射客戶程序:main函數

/* tcpcli01.c */
#include <stdio.h>
#include <strings.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>

int
main(int argc, char **argv)
{
    int                   sockfd;
    struct sockaddr_in    servaddr;
    
    if(argc != 2)
    {
        printf("usage: tcpcli <IPaddress> ");
        exit(0);
    }
    
    if((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
    {
        perror("socket");
        exit(1);
    }
    
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(9877);
    if(inet_pton(AF_INET, argv[1], &servaddr.sin_addr) < 0)
    {
        perror("inet_pton");    
        exit(1);
    }
    
    if(connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)
    {
        perror("connect");
        exit(1);
    }
    
    str_cli(stdin, sockfd);    /* do it all */

    exit(0);
}

TCP回射客戶程序:str_cli函數

/* str_cli.c */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

void
str_cli(FILE *fp, int sockfd)
{
    char    sendline[4096], recvline[4096];

    while(fgets(sendline, 4096, fp) != NULL)
    {
        writen(sockfd, sendline, strlen(sendline));
        
        if(readline(sockfd, recvline, 4096) == 0)
        {
            printf("str_cli: server terminated prematurely");
            exit(0);
        }
        fputs(recvline, stdout);
    }
}

服務器和客戶都要調用的自定義函數:

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>

ssize_t                    /* read "n" bytes from a descriptor. */
readn(int fd, void *vptr, size_t n)
{
    size_t     nleft;
    ssize_t    nread;
    char       *ptr;
    
    ptr = vptr;
    nleft = n;
    while(nleft > 0)
    {
        if((nread = read(fd, ptr, nleft)) < 0)
        {
            if(errno == EINTR)
                nread = 0;    /* and call read() again */
            else
                return(-1);
        }
        else if(nread == 0)
            break;            /* EOF */

        nleft -= nread;
        ptr += nread;
    }
    return(n - nleft);            /* return >= 0 */
}

ssize_t                    /* write n bytes to a descriptor */
writen(int fd, const void *vptr, size_t n)
{
    size_t         nleft;
    ssize_t        nwritten;
    const char    *ptr;

    ptr = vptr;
    nleft = n;
    while(nleft > 0)
    {
        if((nwritten = write(fd, ptr, nleft)) <= 0)
        {
            if(nwritten < 0 && errno == EINTR)
                nwritten = 0;    /* and call write again */
            else
                return(-1);    /* error */
        }
        
        nleft -= nwritten;
        ptr += nwritten;
    }
    return(n - nwritten);
}

ssize_t
readline(int fd, void *vptr, size_t maxlen)
{
    ssize_t        n, rc;
    char           c, *ptr;
    
    ptr = vptr;
    for(n = 1; n < maxlen; n++)
    {
    again:
        if((rc = read(fd, &c, 1)) == 1)
        {
            *ptr++ = c;
            if(c == '\n')
                break;    /* newline is stored, like fgets() */
        }
        else if(rc == 0)
        {
            *ptr = 0;
            return(n - 1);    /* EOF, n - 1 bytes were read */
        }
        else
        {
            if(errno == EINTR)
                goto again;
            return(-1);    /* error, errno set by read() */
        }
    }

    *ptr = 0;    /* null terminate like fgets() */
    return(n);
}

正常啟動:

首先,我們在主機Linux上后台啟動服務器。

image

服務器啟動后,它調用socket、bind、listen和accept,並阻塞於accept調用。(我們還沒有啟動客戶。)在啟動客戶之前,我們運行netstat程序來檢查服務器監聽套接口的狀態。netstat -a

image

這個輸出正是我們所期望的:套接口處於LISTEN狀態,它有通配的本地IP地址,本地端口號為9877(這正是我們所配置的端口號)。netstat用星號“*”來表示一個為0的IP地址(INADDR_ANY,通配地址)或為0的端口號。

我們接着在同一個主機上啟動客戶,並指定服務器主機的IP地址為127.0.0.1(回饋地址)。當然我們也可以指定該地址為 該主機的普通(非回饋)IP地址。

image

客戶調用socket和connect,后者引起TCP的三路握手過程。當三路握手完成后,客戶中的connect和服務器中的accept均返回,連接於是建立。

服務器父進程再次調用accept並阻塞,等待下一個客戶連接。

我們特意在同一個主機上運行客戶和服務器,因為這是試驗客戶/服務器應用程序的最簡單方法。既然我們是在同一個主機上運行客戶和服務器,netstat對於所建立的TCP連接給出兩行輸出(下圖紅色框內)。

image

第一個ESTABLISHED行對應於服務器子進程的套接口,因為它的本地端口號是9877;第二個ESTABLISHED行對應於客戶進程的套接口,因為它的本地端口號是54076. 要是我們在不同的主機上運行客戶和服務器,那么客戶主機就只輸出客戶進程的套接口,服務器主機也只輸出兩個服務器進程(一個父進程、一個子進程)的套接口。

正常終止:

至此連接已經建立,我們在客戶的標准輸入中不論鍵入什么,都回射到它的標准輸出中。

image

注:<Ctrl-D>是我們的終端EOF字符

此時如果立即執行netstat命令,我們將看到如下結果:

image

當前連接的客戶端(它的本地端口號為60779)進入了TIME_WAIT狀態,而監聽服務器仍在等待另一個客戶連接。

在服務器子進程終止時,給父進程發送一個SIGCHILD信號。這一點在本例中發生了,但是我們沒有在代碼中捕獲該信號,而 該信號的缺省行為是被忽略。既然父進程未加處理,子進程於是進入僵死狀態(http://www.cnblogs.com/nufangrensheng/p/3509618.html)。我們可以使用ps命令驗證這一點。

image

子進程的狀態現在是Z(表示僵死)。父進程的狀態現在是S(表示為等待某種資源而睡眠)。

我們必須清理僵死進程,這就涉及到UNIX信號的處理。

POSIX信號處理

信號(signal)就是通知某個進程發生了某個事件,有時也稱為軟件中斷(software interrupt)。信號通常是異步發生的,也就是說進程預先不知道信號准確發生的時刻。

信號可以:

(1)由一個進程發給另一個進程(或自身)。

(2)由內核發給某個進程。

SIGCHILD信號就是由內核在任何一個進程終止時發給它的父進程的一個信號。

每個信號都有一個與之關聯的處置(disposition),也稱為行為(action)。我們調用sigaction函數(http://www.cnblogs.com/nufangrensheng/p/3515945.html)或signal函數(http://www.cnblogs.com/nufangrensheng/p/3514547.html)來設定一個信號的處置,並有三種選擇。

(1)我們可以提供一個函數,它將在特定信號發生的任何時刻被調用。這樣的函數稱為信號處理函數(signal handler),這種行為稱為捕獲(catching)信號。有兩個信號不能被捕獲,它們是SIGKILL和SIGSTOP。信號處理函數由信號值這個單一的整數參數來調用,且沒有返回值,其函數原型如下:

void handler( int signo );

對於大多數信號來說,調用sigaction函數並指定信號發生時所調用的函數就是捕獲信號所要做的全部工作。不過,SIGIO、SIGPOLL和SIGURG這些個別信號還要求捕獲它們的進程做些額外工作。

(2)我們可以把某個信號的處置設定為SIG_IGN來忽略(ignore)它。SIGKILL和SIGSTOP這兩個信號不能被忽略。

(3)我們可以把某個信號的處置設定為SIG_DFL來啟用它的缺省(default)處置。缺省處置通常是在收到信號后終止進程,其中某些信號還在當前工作目錄產生一個進程的核心映像(core image,也稱為內存映像)。另有個別的缺省處理是忽略:SIGCHLD和SIGURG(帶外數據到達時發送)就是缺省處置為忽略的兩個信號。

POSIX信號語義

我們把符合POSIX的系統上的信號處理總結如下:

(1)一旦安裝了信號處理函數,它便一直安裝着(較早期的系統是每執行一次就將其拆除)。

(2)在一個信號處理函數運行期間,正被遞交的信號是阻塞的。而且,安裝處理函數時在傳遞給sigaction函數的sa_mask信號集中指定的任何額外信號也被阻塞。

(3)如果一個信號在被阻塞期間產生了一次或多次,那么該信號被解阻塞之后通常只遞交一次,也就是說UNIX信號缺省是不排隊的。

(4)利用siagprocmask函數(http://www.cnblogs.com/nufangrensheng/p/3515257.html)選擇性地阻塞或解阻塞一組信號是可能的。這使得我們可以做到在一段臨界區代碼執行期間,防止捕獲某些信號,以此保護這段代碼。

處理SIGCHILD信號

設置僵死(zombie)狀態的目的是維護子進程的信息,以便父進程在以后某個時候獲取。這些信息包括子進程的進程ID、終止狀態以及資源利用信息(CPU時間、內存使用量等等)。如果一個進程終止,而該進程有子進程處於僵死狀態,那么它的所有僵死子進程的父進程ID將被重置為1(init進程)。繼承這些子進程的init進程將清理它們(也就是說init進程將wait它們,從而去除它們的僵死狀態)。有些UNIX系統在ps命令輸出的COMMAND欄以<defunct>指明僵死進程。

處理僵死進程

我們顯然不願意留存僵死進程。它們占用內核中的空間,最終可能導致我們耗盡進程資源。無論何時我們fork子進程都得wait它們,以防它們變成僵死進程。為此我們建立一個俘獲SIGCHLD信號的信號處理函數,在函數體中我們調用wait。通過在TCP回射服務器程序:main函數中的listen調用之后增加如下函數調用:

signal(SIGCHLD, sig_chld);

這樣我們就建立了該信號處理函數。(這必須在fork第一個子進程之前完成,並且只做一次。) 我們接着定義名為sig_chld的這個信號處理函數,如下:

void 
sig_chld(int signo)
{
    pid_t pid;
    int   stat;

    pid = wait(&stat);
    printf("child %d terminatted\n", pid);
    return;
}

處理僵死進程的可移植方法就是捕獲SIGCHLD,並調用wait或waitpid。

新的問題是:在某些系統上(這些 系統標准C函數庫中提供的signal函數不會致使內核自動重啟被中斷的系統調用),SIGCHLD信號被捕獲並處理后,慢系統調用accept會返回一個EINTR錯誤(被中斷的系統調用).

處理被中斷的系統調用

慢系統調用(slow system call)是指那些可能永遠阻塞的系統調用(調用有可能永遠無法返回)。多數網絡支持函數都屬於這一類。

適用於慢系統調用的基本規則是:當阻塞於某個慢系統調用的一個進程捕獲某個信號且相應信號處理函數返回時,該系統調用可能返回一個EINTR錯誤。有些內核自動重啟某些被中斷的系統調用。

為了處理被中斷的accept,我們把對accept的調用從for循環開始修改如下:

for(;;)
{
    clilen = sizeof(cliaddr);
    if((connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &clilen)) < 0)
    {
        if(errno == EINTR)
            continue;    /* back to for() */
        else
        {
            perror("accept error");
            exit(1);
        }
    }
}

這段代碼所做的事情就是自己重啟被中斷的系統調用。對於accept以及諸如read、write、select和open之類函數來說,這是合適的。不過有一個函數我們不能重啟:connect。如果該函數返回EINTR,我們就不能再次調用它,否則將立即返回一個錯誤。當connect被一個捕獲的信號中斷而且不自動重啟時,我們必須調用select來等待連接完成。

wait和waitpid函數

#include <sys/wait.h>

pid_t wait(int *statloc);

pid_t waitpid(pid_t pid, int *statloc, int options);

二者均返回:若成功則返回已終止子進程的進程ID,若出錯則返回-1

函數wait和waitpid均返回兩個值:函數返回值是已終止子進程ID號,子進程的終止狀態(一個整數)則通過statloc指針返回。我們可以調用三個宏來檢查終止狀態,並辨別子進程是正常終止、由某個信號殺死還是僅僅由作業控制停止而已(http://www.cnblogs.com/nufangrensheng/p/3510101.html)。

如果調用wait的進程沒有已終止的子進程,不過有一個或多個子進程仍在執行,那么wait將阻塞到現有子進程第一個終止為止。

waitpid函數對於等待哪個進程以及是否阻塞給了我們更多的控制。首先,pid參數允許我們指定想等待的進程ID,值-1表示等待第一個終止的子進程。其次,options參數允許我們指定附加選項。最常用的選項是WNOHANG,它告知內核在沒有已終止子進程時不要阻塞。

函數wait和waipid的區別

為了說明wait和waitpid的區別,我們試想如下情況:

image

當客戶終止時,所有打開的描述符字由內核自動關閉,且所有5個連接基本在同一時刻終止。這就引發了5個FIN,每個連接一個,它們反過來使服務器的5個子進程基本在同一時刻終止。這又導致差不多在同一時刻遞交5個SIGCHLD信號給父進程。

image

如果我們調用函數wait來處理已終止的子進程,那么只會捕獲到一個SIGCHLD信號。也就是說,其他的4個子進程仍然作為僵死進程存在着。

所以,建立一個信號處理函數並在其中調用wait並不足以防止出現僵死進程。本問題在於:所有5個信號都在信號處理函數執行之前產生,而信號處理函數只執行一次,因為UNIX信號一般是不排隊的。

正確的解決辦法是調用waitpid而不是wait。如下所示給出了正確處理SIGCHLD的sig_chld函數的版本。這個版本管用的原因在於:我們在一個循環內調用waitpid,以獲取所有已終止子進程的狀態。我們必須指定WNOHANG選項,它告知waitpid在有尚未終止的子進程在運行時不要阻塞。我們不能在循環內調用wait,因為沒有辦法防止wait在尚有未終止的子進程在運行時阻塞。

void
sig_chld(int signo)
{
    pid_t    pid;
    int      stat;

    while((pid = waitpid(-1, &stat, WNOHANG)) > 0)
        printf("child %d terminated\n", pid);
    return;
}

服務器的最終版本

/* tcpserv01.c */
#include <sys/socket.h>
#include <strings.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>

int
main(int argc, char **argv)
{
    int                    listenfd, connfd;
    pid_t                  childpid;
    socklen_t              clilen;
    struct sockaddr_in     cliaddr, servaddr;
    void                   sig_chld(int);
    
    if((listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
    {
        perror("socket");
        exit(1);
    }
    
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(9877);

    if(bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)
    {
        perror("bind");
        exit(1);
    }
    if(listen(listenfd, 5) < 0)
    {
        perror("listen");
        exit(1);
    }
    
   signal(SIGCHLD, sig_chld);    /* must call waitpid() */
    for(;;)
    {
        clilen = sizeof(cliaddr);
        if((connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &clilen)) < 0)
        {
            if(errno == EINTR)
                  continue;        /* back to for */        
            else
            {
                  perror("accept");
                  exit(1);
            }
        }
        
        if((childpid = fork()) < 0)
        {
            perror("fork");
            exit(1);
        }
        else if(childpid == 0)    /* child process */
        {
            if(close(listenfd) < 0) /* close listening socket */
            {
                perror("child close");
                exit(1);
            }
            str_echo(connfd);    /* process the request */
            exit(0);
            
        }
        if(close(connfd) < 0) /* parent close connected socket */
        {
            perror("parent close");
            exit(1);
        }
    }
    
    
}

在網絡編程時可能會遇到的三種情況:

(1)當fork子進程時,必須捕獲SIGCHLD信號。

(2)當捕獲信號時,必須處理被中斷的系統調用。

(3)SIGCHLD的信號處理函數必須正確編寫,應使用waitpid函數以免留下僵死進程。

TCP程序實例小結:

在客戶和服務器可以彼此通信之前,每一端都得指定連接的套接口對:本地IP地址、本地端口、遠地IP地址、遠地端口。在下圖中我們以粗體圓點標出了這四個值。該圖處於客戶端的角度。遠地IP地址和遠地端口必須在客戶端調用connect時指定,而兩個本地值通常就由內核作為connect的一部分來選定。客戶也可在調用connect之前,通過調用bind來指定其中一個或全部兩個本地值,不過這么做不常見。

image

客戶可以在連接建立后通過調用getsockname獲取由內核指定的兩個本地值。

下圖中標出了同樣的四個值,不過是處於服務器的角度。

本地端口(服務器眾所周知的端口)由bind指定。bind調用中指定的本地IP地址通常是通配IP地址,盡管服務器也可以指定一個非通配的IP地址來限定接收目標為某個特定本地接口的連接。如果服務器在一個多宿主機上綁定通配IP地址,那么它可以在連接建立后通過調用getsockname來確定本地IP地址。兩個遠地值則由accept調用返回給服務器。如果另外一個程序由調用accept的服務器通過調用exec來執行,則這個新程序可以在必要時調用getpeername來確定客戶的IP地址和端口號。

image

數據格式

例子:在客戶和服務器之間 傳遞文本串

修改我們的服務器程序,它仍然從客戶讀入一行文本,不過新的服務器期望該文本行包含由空格分開的兩個整數,服務器將返回這兩個整數的和。我們的客戶和服務器程序的main函數仍保持不變,str_cli函數也保持不變,所有修改都在str_echo函數中,如下所示:

#include <stdio.h>
#include <sys/types.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

void
str_echo(int sockfd)
{
    long       arg1, arg2;
    ssize_t    n;
    char       line[4096];

    for(;;)
    {
        if((n = readline(sockfd, line, 4096)) == 0)
            return;    /*  connection closed by other end */
        if(sscanf(line, "%ld%ld", &arg1, &arg2) == 2)
            snprintf(line, sizeof(line), "%ld\n", arg1 + arg2);
        else
            snprintf(line, sizeof(line), "input error\n");

        n = strlen(line);
        writen(sockfd, line, n);
    }
}

我們調用sscanf把文本串中的兩個參數轉換為長整數,然后調用snprintf把結果轉換為文本串。

不論客戶和服務器主機的字節序如何,這個新的客戶和服務器對都工作的很好。

例子:在客戶與服務器之間傳遞二進制結構

現在我們 把客戶和服務器程序修改為穿越套接口傳遞二進制值而不是文本串。

我們的客戶和服務器程序的main函數無需改動。另外我們給兩個參數定義了一個結構,給結果定義了另一個結構。

#ifndef _COMMON_H
#define _COMMON_H

struct args
{
    long arg1;
    long arg2;    
};
struct result
{
    long sum;
};

#endif
/* str_cli09.c */
#include "common.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

void
str_cli(FILE *fp, int sockfd)
{
    char    sendline[4096], recvline[4096];
    struct args    args;    
    struct result    result;

    while(fgets(sendline, 4096, fp) != NULL)
    {
        if(sscanf(sendline, "%ld%ld", &args.arg1, &args.arg2) != 2)
        {
            printf("invalid input: %s", sendline);
            continue;
        }
        writen(sockfd, &args, sizeof(args));

        if(readn(sockfd, &result, sizeof(result)) == 0)
        {
            printf("str_cli: server terminated prematurely");
            exit(1);
        }
        
        printf("%ld\n", result.sum);
    }
}
/* str_echo09.c */
#include <stdio.h>
#include "common.h"
#include <stdlib.h>
#include <errno.h>

void
str_echo(int sockfd)
{
    ssize_t        n;
    struct args    args;
    struct result    result;
    
    for(;;)
    {
        if((n = readn(sockfd, &args, sizeof(args))) == 0)
            return;
        result.sum = args.arg1 + args.arg2;
        writen(sockfd, &result, sizeof(result));
    }
    
}

如果我們在具有相同體系結構的兩個主機上運行我們的客戶和服務器程序,那么什么問題都沒有。但是如果在具有不同體系結構的兩個主機上運行同樣的客戶和服務器程序(例如服務器運行在大端系統,而客戶運行在小端系統上),那就無法工作了。

本例子實際上存在三個潛在的問題:

(1)不同的實現以不同的格式存儲二進制數。(大端和小端)

(2)不同的實現在存儲相同的C數據類型上可能存在差異。(32位系統和64位系統)

(3)不同的實現給結構打包的方式存在差異,這取決於各種數據類型所用的位數以及機器的對齊限制。

因此,穿越套接口傳送二進制結構絕不是明智的選擇。

解決這種數據格式問題有兩個常用的方法:

(1)把所有的數值數據作為文本串來傳遞。當然這里假設客戶和服務器主機具有相同的字符集。

(2)顯示定義所支持數據類型的二進制格式(位數、大端或小端),並以這樣的格式在客戶和服務器之間傳遞所有數據。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM