您好,登錄后才能下訂單哦!
在 TCP/IP 協(xié)議中,"IP地址 + TCP或UDP端口號(hào)" 可以唯一標(biāo)識(shí)網(wǎng)絡(luò)通訊中的一個(gè)進(jìn)程,"IP地址+端口號(hào)" 就稱為 socket。本文以一個(gè)簡(jiǎn)單的 TCP 協(xié)議為例,介紹如何創(chuàng)建基于 TCP 協(xié)議的網(wǎng)絡(luò)程序。
TCP 協(xié)議通訊流程
下圖描述了 TCP 協(xié)議的通訊流程(此圖來自互聯(lián)網(wǎng)):
下圖則描述 TCP 建立連接的過程(此圖來自互聯(lián)網(wǎng)):
服務(wù)器調(diào)用 socket()、bind()、listen() 函數(shù)完成初始化后,調(diào)用 accept() 阻塞等待,處于監(jiān)聽端口的狀態(tài),客戶端調(diào)用 socket() 初始化后,調(diào)用 connect() 發(fā)出 SYN 段并阻塞等待服務(wù)器應(yīng)答,服務(wù)器應(yīng)答一個(gè)SYN-ACK 段,客戶端收到后從 connect() 返回,同時(shí)應(yīng)答一個(gè) ACK 段,服務(wù)器收到后從 accept() 返回。
TCP 連接建立后數(shù)據(jù)傳輸?shù)倪^程:
建立連接后,TCP 協(xié)議提供全雙工的通信服務(wù),但是一般的客戶端/服務(wù)器程序的流程是由客戶端主動(dòng)發(fā)起請(qǐng)求,服務(wù)器被動(dòng)處理請(qǐng)求,一問一答的方式。因此,服務(wù)器從 accept() 返回后立刻調(diào)用 read(),讀 socket 就像讀管道一樣,如果沒有數(shù)據(jù)到達(dá)就阻塞等待,這時(shí)客戶端調(diào)用 write() 發(fā)送請(qǐng)求給服務(wù)器,服務(wù)器收到后從 read() 返回,對(duì)客戶端的請(qǐng)求進(jìn)行處理,在此期間客戶端調(diào)用 read() 阻塞等待服務(wù)器的應(yīng)答,服務(wù)器調(diào)用 write() 將處理結(jié)果發(fā)回給客戶端,再次調(diào)用 read() 阻塞等待下一條請(qǐng)求,客戶端收到后從 read() 返回,發(fā)送下一條請(qǐng)求,如此循環(huán)下去。
下圖描述了關(guān)閉 TCP 連接的過程:
如果客戶端沒有更多的請(qǐng)求了,就調(diào)用 close() 關(guān)閉連接,就像寫端關(guān)閉的管道一樣,服務(wù)器的 read() 返回 0,這樣服務(wù)器就知道客戶端關(guān)閉了連接,也調(diào)用 close() 關(guān)閉連接。注意,任何一方調(diào)用 close() 后,連接的兩個(gè)傳輸方向都關(guān)閉,不能再發(fā)送數(shù)據(jù)了。如果一方調(diào)用 shutdown() 則連接處于半關(guān)閉狀態(tài),仍可接收對(duì)方發(fā)來的數(shù)據(jù)。
在學(xué)習(xí) socket 編程時(shí)要注意應(yīng)用程序和 TCP 協(xié)議層是如何交互的:
下面通過一個(gè)簡(jiǎn)單的 TCP 網(wǎng)絡(luò)程序來理解相關(guān)概念。程序分為服務(wù)器端和客戶端兩部分,它們之間通過 socket 進(jìn)行通信。
服務(wù)器端程序
下面是一個(gè)非常簡(jiǎn)單的服務(wù)器端程序,它從客戶端讀字符,然后將每個(gè)字符轉(zhuǎn)換為大寫并回送給客戶端:
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <netinet/in.h> #define MAXLINE 80 #define SERV_PORT 8000 int main(void) { struct sockaddr_in servaddr, cliaddr; socklen_t cliaddr_len; int listenfd, connfd; char buf[MAXLINE]; char str[INET_ADDRSTRLEN]; int i, n; // socket() 打開一個(gè)網(wǎng)絡(luò)通訊端口,如果成功的話, // 就像 open() 一樣返回一個(gè)文件描述符, // 應(yīng)用程序可以像讀寫文件一樣用 read/write 在網(wǎng)絡(luò)上收發(fā)數(shù)據(jù)。 listenfd = socket(AF_INET, SOCK_STREAM, 0); bzero(&servaddr, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = htonl(INADDR_ANY); servaddr.sin_port = htons(SERV_PORT); // bind() 的作用是將參數(shù) listenfd 和 servaddr 綁定在一起, // 使 listenfd 這個(gè)用于網(wǎng)絡(luò)通訊的文件描述符監(jiān)聽 servaddr 所描述的地址和端口號(hào)。 bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)); // listen() 聲明 listenfd 處于監(jiān)聽狀態(tài), // 并且最多允許有 20 個(gè)客戶端處于連接待狀態(tài),如果接收到更多的連接請(qǐng)求就忽略。 listen(listenfd, 20); printf("Accepting connections ...\n"); while (1) { cliaddr_len = sizeof(cliaddr); // 典型的服務(wù)器程序可以同時(shí)服務(wù)于多個(gè)客戶端, // 當(dāng)有客戶端發(fā)起連接時(shí),服務(wù)器調(diào)用的 accept() 返回并接受這個(gè)連接, // 如果有大量的客戶端發(fā)起連接而服務(wù)器來不及處理,尚未 accept 的客戶端就處于連接等待狀態(tài)。 connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &cliaddr_len); n = read(connfd, buf, MAXLINE); printf("received from %s at PORT %d\n", inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)), ntohs(cliaddr.sin_port)); for (i = 0; i < n; i++) { buf[i] = toupper(buf[i]); } write(connfd, buf, n); close(connfd); } }
把上面的代碼保存到文件 server.c 文件中,并執(zhí)行下面的命令編譯:
$ gcc server.c -o server
然后運(yùn)行編譯出來的 server 程序:
$ ./server
此時(shí)我們可以通過 ss 命令來查看主機(jī)上的端口監(jiān)聽情況:
如上圖所示,server 程序已經(jīng)開始監(jiān)聽主機(jī)的 8000 端口了。
下面讓我們介紹一下這段程序中用到的 socket 相關(guān)的 API。
int socket(int family, int type, int protocol);
socket() 打開一個(gè)網(wǎng)絡(luò)通訊端口,如果成功的話,就像 open() 一樣返回一個(gè)文件描述符,應(yīng)用程序可以像讀寫文件一樣用 read/write 在網(wǎng)絡(luò)上收發(fā)數(shù)據(jù)。對(duì)于IPv4,family 參數(shù)指定為 AF_INET。對(duì)于 TCP 協(xié)議,type 參數(shù)指定為 SOCK_STREAM,表示面向流的傳輸協(xié)議。如果是 UDP 協(xié)議,則 type 參數(shù)指定為 SOCK_DGRAM,表示面向數(shù)據(jù)報(bào)的傳輸協(xié)議。protocol 指定為 0 即可。
int bind(int sockfd, const struct sockaddr *myaddr, socklen_t addrlen);
服務(wù)器需要調(diào)用 bind 函數(shù)綁定一個(gè)固定的網(wǎng)絡(luò)地址和端口號(hào)。bind() 的作用是將參數(shù) sockfd 和 myaddr 綁定在一起,使 sockfd 這個(gè)用于網(wǎng)絡(luò)通訊的文件描述符監(jiān)聽 myaddr 所描述的地址和端口號(hào)。struct sockaddr *是一個(gè)通用指針類型,myaddr 參數(shù)實(shí)際上可以接受多種協(xié)議的 sockaddr 結(jié)構(gòu)體,而它們的長(zhǎng)度各不相同,所以需要第三個(gè)參數(shù) addrlen 指定結(jié)構(gòu)體的長(zhǎng)度。
程序中對(duì) myaddr 參數(shù)的初始化為:
bzero(&servaddr, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = htonl(INADDR_ANY); servaddr.sin_port = htons(SERV_PORT);
首先將整個(gè)結(jié)構(gòu)體清零,然后設(shè)置地址類型為 AF_INET,網(wǎng)絡(luò)地址為 INADDR_ANY,這個(gè)宏表示本地的任意 IP 地址,因?yàn)榉?wù)器可能有多個(gè)網(wǎng)卡,每個(gè)網(wǎng)卡也可能綁定多個(gè) IP 地址,這樣設(shè)置可以在所有的 IP 地址上監(jiān)聽,直到與某個(gè)客戶端建立了連接時(shí)才確定下來到底用哪個(gè) IP 地址,端口號(hào)為 SERV_PORT,我們定義為 8000。
int listen(int sockfd, int backlog);
listen() 聲明 sockfd 處于監(jiān)聽狀態(tài),并且最多允許有 backlog 個(gè)客戶端處于連接待狀態(tài),如果接收到更多的連接請(qǐng)求就忽略。
int accept(int sockfd, struct sockaddr *cliaddr, socklen_t *addrlen);
三方握手完成后,服務(wù)器調(diào)用 accept() 接受連接,如果服務(wù)器調(diào)用 accept() 時(shí)還沒有客戶端的連接請(qǐng)求,就阻塞等待直到有客戶端連接上來。cliaddr 是一個(gè)傳出參數(shù),accept() 返回時(shí)傳出客戶端的地址和端口號(hào)。addrlen 參數(shù)是一個(gè)傳入傳出參數(shù)(value-result argument),傳入的是調(diào)用者提供的緩沖區(qū) cliaddr 的長(zhǎng)度以避免緩沖區(qū)溢出問題,傳出的是客戶端地址結(jié)構(gòu)體的實(shí)際長(zhǎng)度(有可能沒有占滿調(diào)用者提供的緩沖區(qū))。如果給 cliaddr 參數(shù)傳 NULL,表示不關(guān)心客戶端的地址。
服務(wù)器程序的主要結(jié)構(gòu)如下:
while (1) { cliaddr_len = sizeof(cliaddr); connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &cliaddr_len); n = read(connfd, buf, MAXLINE); ...... close(connfd); }
整個(gè)是一個(gè) while 死循環(huán),每次循環(huán)處理一個(gè)客戶端連接。由于 cliaddr_len 是傳入傳出參數(shù),每次調(diào)用 accept( ) 之前應(yīng)該重新賦初值。accept() 的參數(shù) listenfd 是先前的監(jiān)聽文件描述符,而 accept() 的返回值是另外一個(gè)文件描述符 connfd,之后與客戶端之間就通過這個(gè) connfd 通訊,最后關(guān)閉 connfd 斷開連接,而不關(guān)閉 listenfd,再次回到循環(huán)開頭 listenfd 仍然用作 accept 的參數(shù)。
客戶端程序
下面是客戶端程序,它從命令行參數(shù)中獲得一個(gè)字符串發(fā)給服務(wù)器,然后接收服務(wù)器返回的字符串并打?。?/p>
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <netinet/in.h> #include <arpa/inet.h> #define MAXLINE 80 #define SERV_PORT 8000 int main(int argc, char *argv[]) { struct sockaddr_in servaddr; char buf[MAXLINE]; int sockfd, n; char *str; if (argc != 2) { fputs("usage: ./client message\n", stderr); exit(1); } str = argv[1]; sockfd = socket(AF_INET, SOCK_STREAM, 0); bzero(&servaddr, sizeof(servaddr)); servaddr.sin_family = AF_INET; inet_pton(AF_INET, "127.0.0.1", &servaddr.sin_addr); servaddr.sin_port = htons(SERV_PORT); // 由于客戶端不需要固定的端口號(hào),因此不必調(diào)用 bind(),客戶端的端口號(hào)由內(nèi)核自動(dòng)分配。 // 注意,客戶端不是不允許調(diào)用 bind(),只是沒有必要調(diào)用 bind() 固定一個(gè)端口號(hào), // 服務(wù)器也不是必須調(diào)用 bind(),但如果服務(wù)器不調(diào)用 bind(),內(nèi)核會(huì)自動(dòng)給服務(wù)器分配監(jiān)聽端口, // 每次啟動(dòng)服務(wù)器時(shí)端口號(hào)都不一樣,客戶端要連接服務(wù)器就會(huì)遇到麻煩。 connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr)); write(sockfd, str, strlen(str)); n = read(sockfd, buf, MAXLINE); printf("Response from server:\n"); write(STDOUT_FILENO, buf, n); printf("\n"); close(sockfd); return 0; }
把上面的代碼保存到文件 client.c 文件中,并執(zhí)行下面的命令編譯:
$ gcc client.c -o client
然后運(yùn)行編譯出來的 client 程序:
$ ./client hello
此時(shí)服務(wù)器端會(huì)收到請(qǐng)求并返回轉(zhuǎn)換為大寫的字符串,并輸出相應(yīng)的信息:
而客戶端在發(fā)送請(qǐng)求后會(huì)收到轉(zhuǎn)換過的字符串:
在客戶端的代碼中有兩點(diǎn)需要注意:
1. 由于客戶端不需要固定的端口號(hào),因此不必調(diào)用 bind(),客戶端的端口號(hào)由內(nèi)核自動(dòng)分配。
2. 客戶端需要調(diào)用 connect() 連接服務(wù)器,connect 和 bind 的參數(shù)形式一致,區(qū)別在于 bind 的參數(shù)是自己的地址,而 connect 的參數(shù)是對(duì)方的地址。
至此我們已經(jīng)使用 socket 技術(shù)完成了一個(gè)最簡(jiǎn)單的客戶端服務(wù)器程序,雖然離實(shí)際應(yīng)用還非常遙遠(yuǎn),但就學(xué)習(xí)而言已經(jīng)足夠了。
提升服務(wù)器端的響應(yīng)能力
雖然我們的服務(wù)器程序可以響應(yīng)客戶端的請(qǐng)求,但是這樣的效率太低了。一般情況下服務(wù)器程序需要能夠同時(shí)處理多個(gè)客戶端的請(qǐng)求??梢酝ㄟ^ fork 系統(tǒng)調(diào)用創(chuàng)建子進(jìn)程來處理每個(gè)請(qǐng)求,下面是大體的實(shí)現(xiàn)思路:
listenfd = socket(...); bind(listenfd, ...); listen(listenfd, ...); while (1) { connfd = accept(listenfd, ...); n = fork(); if (n == -1) { perror("call to fork"); exit(1); } else if (n == 0) { // 在子進(jìn)程中處理客戶端的請(qǐng)求。 close(listenfd); while (1) { read(connfd, ...); ... write(connfd, ...); } close(connfd); exit(0); } else { close(connfd); } }
此時(shí)父進(jìn)程的任務(wù)就是不斷的創(chuàng)建子進(jìn)程,而由子進(jìn)程去響應(yīng)客戶端的具體請(qǐng)求。通過這種方式,可以極大的提升服務(wù)器端的響應(yīng)能力。
總結(jié)
本文通過一個(gè)簡(jiǎn)單的建基于 TCP 協(xié)議的網(wǎng)絡(luò)程序介紹了 linux socket 編程中的基本概念。通過它我們可以了解到 socket 程序工作的基本原理,以及一些解決性能問題的思路。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。