該節(jié)主要展示一個簡單的[日期時間]服務器的程序示例.
§1.3 一個簡單的日期時間服務器程序代碼
???這個服務器程序可以為上一節(jié)的客戶端提供服務。
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 }
產(chǎn)生一個TCP套接字
10
創(chuàng)建一個TCP套接字,與客戶端一樣。
綁定服務器的廣為人知的端口到該套接字
11–15
服務器通過填充網(wǎng)絡套接字結(jié)構體中的端口域,以及服務器的網(wǎng)絡接口(IP地址),然后進行綁定(調(diào)用bind)。在這里指定IP地址為INADDR_ANY,為了讓客戶端可以連接服務器的任一網(wǎng)絡接口(因為服務器可能有多塊網(wǎng)卡,也就對應了多個IP地址),也就是說如果服務器有兩個IP地址,客戶端連接任一IP地址即可。后續(xù)章節(jié)中介紹了如何限制客戶端連接到一個固定的接口上。
轉(zhuǎn)換為監(jiān)聽套接字
16
通過調(diào)用listen,一個套接字就轉(zhuǎn)換為監(jiān)聽套接字,這就是說該套接字負責接收來自客戶端的連接請求,而并不真正與客戶端進行信息傳輸。
常量LISTENQ 是在頭文件unp.h中定義的,它是指能夠同時監(jiān)聽客戶端連接的個數(shù)。不超過LISTENQ的客戶端同時連接服務器,它們會在一個隊列中排隊,來等待服務器的處理。后續(xù)章節(jié)有更詳細的討論。
接收客戶端連接,發(fā)送回復
17–21
一般地,服務器進程在調(diào)用accept之后進入到睡眠狀態(tài),等待著客戶端地連接請求. 一個TCP連接通過一個稱為三方握手來建立,當三方握手完成之后,accept調(diào)用返回。返回值是一個新的套接字描述符(一個整數(shù)值connfd),這個新的套接字負責與客戶端進行通訊。對于每一個客戶端地連接,accept都返回一個新的套接字描述符。整本書使用的無限循環(huán)風格是這樣的:
for ( ; ; ) {
. . .
}
當前時間和日期通過調(diào)用庫函數(shù)time來獲得,并且通過調(diào)用ctime進行轉(zhuǎn)換,使得我們能夠直觀的閱讀。如下:
Mon May 26 20:58:40 2003
終止連接
22
客戶端調(diào)用close之后,服務器關閉連接。這時候引起了一個TCP連接終止序列:一個FIN發(fā)送到每一端,同時每一個FIN都要被另一端確認。在后面章節(jié)中將會對TCP連接建立時候的三方握手以及TCP連接終止時候的四包交換有更詳細的討論。
???以上給出的客戶端和服務器版本都是協(xié)議相關的(IPv4),在后面將會給出一個協(xié)議無關的版本(IPv4和IPv6都適用,主要通過使用getaddrinfo函數(shù))。
???最后需要補充的一點是,在以上涉及到Socket API調(diào)用的時候,每個函數(shù)的第一個字母變成了大寫,其意義和小寫開頭的是一樣的,只不過多了一個錯誤處理罷了。