socket 套接字理解

socket

我們知道兩個進程如果需要進行通訊最基本的一個前提能能夠唯一的標示一個進程,在本地進程通訊中我們可以使用PID來唯一標示一個進程,但PID只在本地唯一,網絡中的兩個進程PID衝突機率很大,這時候我們需要另闢它徑了,我們知道IP層的ip地址可以唯一標示主機,而TCP層協議和端口號可以唯一標示主機的一個進程,這樣我們可以利用ip地址+協議+端口號唯一標示網絡中的一個進程。

能夠唯一標示網絡中的進程後,它們就可以利用socket進行通信了,什麼是socket呢?我們經常把socket翻譯爲套接字,socket是在應用層和傳輸層之間的一個抽象層,它把TCP/IP層複雜的操作抽象爲幾個簡單的接口供應用層調用已實現進程在網絡中通信。

socket通信流程

socket是"打開—讀/寫—關閉"模式的實現,以使用TCP協議通訊的socket爲例,其交互流程大概是這樣子的

服務器根據地址類型(ipv4,ipv6)、socket類型、協議創建socket

服務器爲socket綁定ip地址和端口號

服務器socket監聽端口號請求,隨時準備接收客戶端發來的連接,這時候服務器的socket並沒有被打開

客戶端創建socket

客戶端打開socket,根據服務器ip地址和端口號試圖連接服務器socket

服務器socket接收到客戶端socket請求,被動打開,開始接收客戶端請求,直到客戶端返回連接信息。這時候socket進入阻塞狀態,所謂阻塞即accept()方法一直到客戶端返回連接信息後才返回,開始接收下一個客戶端諒解請求

客戶端連接成功,向服務器發送連接狀態信息

服務器accept方法返回,連接成功

客戶端向socket寫入信息

服務器讀取信息

客戶端關閉

服務器端關閉

三次握手

在TCP/IP協議中,TCP協議通過三次握手建立一個可靠的連接

第一次握手:客戶端嘗試連接服務器,向服務器發送syn包(同步序列編號Synchronize Sequence Numbers),syn=j,客戶端進入SYN_SEND狀態等待服務器確認

第二次握手:服務器接收客戶端syn包並確認(ack=j+1),同時向客戶端發送一個SYN包(syn=k),即SYN+ACK包,此時服務器進入SYN_RECV狀態

第三次握手:第三次握手:客戶端收到服務器的SYN+ACK包,向服務器發送確認包ACK(ack=k+1),此包發送完畢,客戶端和服務器進入ESTABLISHED狀態,完成三次握手

定睛一看,服務器socket與客戶端socket建立連接的部分其實就是大名鼎鼎的三次握手

 socket編程API

前面提到socket是"打開—讀/寫—關閉"模式的實現,簡單瞭解一下socket提供了哪些API供應用程序使用,還是以TCP協議爲例,看看Unix下的socket API,其它語言都很類似(PHP甚至名字都幾乎一樣),這裏我就簡單解釋一下方法作用和參數,具體使用有興趣同學可以看看博客參考中的鏈接或者上網搜索

int socket(int domain, int type, int protocol);

根據指定的地址族、數據類型和協議來分配一個socket的描述字及其所用的資源。

domain:協議族,常用的有AF_INETAF_INET6AF_LOCALAF_ROUTE其中AF_INET代表使用ipv4地址

type:socket類型,常用的socket類型有,SOCK_STREAMSOCK_DGRAMSOCK_RAWSOCK_PACKETSOCK_SEQPACKET

protocol:協議。常用的協議有,IPPROTO_TCPIPPTOTO_UDPIPPROTO_SCTPIPPROTO_TIPC

int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

把一個地址族中的特定地址賦給socket

sockfd:socket描述字,也就是socket引用

addr:要綁定給sockfd的協議地址

addrlen:地址的長度

通常服務器在啓動的時候都會綁定一個衆所周知的地址(如ip地址+端口號),用於提供服務,客戶就可以通過它來接連服務器;而客戶端就不用指定,有系統自動分配一個端口號和自身的ip地址組合。這就是爲什麼通常服務器端在listen之前會調用bind(),而客戶端就不會調用,而是在connect()時由系統隨機生成一個。

int listen(int sockfd, int backlog);

監聽socket

sockfd:要監聽的socket描述字

backlog:相應socket可以排隊的最大連接個數 

int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

連接某個socket

sockfd:客戶端的socket描述字

addr:服務器的socket地址

addrlen:socket地址的長度

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);

 TCP服務器監聽到客戶端請求之後,調用accept()函數取接收請求

sockfd:服務器的socket描述字

addr:客戶端的socket地址

addrlen:socket地址的長度

ssize_t read(int fd, void *buf, size_t count);

讀取socket內容

fd:socket描述字

buf:緩衝區

count:緩衝區長度

ssize_t write(int fd, const void *buf, size_t count);

向socket寫入內容,其實就是發送內容

fd:socket描述字

buf:緩衝區

count:緩衝區長度

int close(int fd);

socket標記爲以關閉使相應socket描述字的引用計數-1,當引用計數爲0的時候,觸發TCP客戶端向服務器發送終止連接請求。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章