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