該節主要展示一個簡單的[日期時間]服務器的程序示例.
§1.3 一個簡單的日期時間服務器程序代碼
???這個服務器程序可以為上一節的客戶端提供服務。
1 #include "unp.h"
2 #include <time.h>
3 int
4 main(int argc, char **argv)
5 {
6 int listenfd, connfd;
7 struct sockaddr_in servaddr;
8 char buff[MAXLINE];
9 time_t ticks;
10 listenfd = Socket(AF_INET, SOCK_STREAM, 0);
11 bzeros(&servaddr, sizeof(servaddr));
12 servaddr.sin_family = AF_INET;
13 servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
14 servaddr.sin_port = htons(13); /* daytime server */
15 Bind(listenfd, (SA *) &servaddr, sizeof(servaddr));
16 Listen(listenfd, LISTENQ);
17 for ( ; ; ) {
18 connfd = Accept(listenfd, (SA *) NULL, NULL);
19 ticks = time(NULL);
20 snprintf(buff, sizeof(buff), "%.24s\r\n", ctime(&ticks));
21 Write(connfd, buff, strlen(buff));
22 Close(connfd);
23 }
24 }
產生一個TCP套接字
10
創建一個TCP套接字,與客戶端一樣。
綁定服務器的廣為人知的端口到該套接字
11–15
服務器通過填充網絡套接字結構體中的端口域,以及服務器的網絡接口(IP地址),然后進行綁定(調用bind)。在這里指定IP地址為INADDR_ANY,為了讓客戶端可以連接服務器的任一網絡接口(因為服務器可能有多塊網卡,也就對應了多個IP地址),也就是說如果服務器有兩個IP地址,客戶端連接任一IP地址即可。后續章節中介紹了如何限制客戶端連接到一個固定的接口上。
轉換為監聽套接字
16
通過調用listen,一個套接字就轉換為監聽套接字,這就是說該套接字負責接收來自客戶端的連接請求,而并不真正與客戶端進行信息傳輸。
常量LISTENQ 是在頭文件unp.h中定義的,它是指能夠同時監聽客戶端連接的個數。不超過LISTENQ的客戶端同時連接服務器,它們會在一個隊列中排隊,來等待服務器的處理。后續章節有更詳細的討論。
接收客戶端連接,發送回復
17–21
一般地,服務器進程在調用accept之后進入到睡眠狀態,等待著客戶端地連接請求. 一個TCP連接通過一個稱為三方握手來建立,當三方握手完成之后,accept調用返回。返回值是一個新的套接字描述符(一個整數值connfd),這個新的套接字負責與客戶端進行通訊。對于每一個客戶端地連接,accept都返回一個新的套接字描述符。整本書使用的無限循環風格是這樣的:
for ( ; ; ) {
. . .
}
當前時間和日期通過調用庫函數time來獲得,并且通過調用ctime進行轉換,使得我們能夠直觀的閱讀。如下:
Mon May 26 20:58:40 2003
終止連接
22
客戶端調用close之后,服務器關閉連接。這時候引起了一個TCP連接終止序列:一個FIN發送到每一端,同時每一個FIN都要被另一端確認。在后面章節中將會對TCP連接建立時候的三方握手以及TCP連接終止時候的四包交換有更詳細的討論。
???以上給出的客戶端和服務器版本都是協議相關的(IPv4),在后面將會給出一個協議無關的版本(IPv4和IPv6都適用,主要通過使用getaddrinfo函數)。
???最后需要補充的一點是,在以上涉及到Socket API調用的時候,每個函數的第一個字母變成了大寫,其意義和小寫開頭的是一樣的,只不過多了一個錯誤處理罷了。