TCP/IP
要想理解socket首先得熟悉一下TCP/IP協(xié)議族世囊, TCP/IP(Transmission Control Protocol/Internet Protocol)即傳輸控制協(xié)議/網(wǎng)間協(xié)議堪澎,定義了主機(jī)如何連入因特網(wǎng)及數(shù)據(jù)如何再它們之間傳輸?shù)臉?biāo)準(zhǔn)济瓢,
從字面意思來(lái)看TCP/IP是TCP和IP協(xié)議的合稱(chēng),但實(shí)際上TCP/IP協(xié)議是指因特網(wǎng)整個(gè)TCP/IP協(xié)議族。不同于ISO模型的七個(gè)分層坡贺,TCP/IP協(xié)議參考模型把所有的TCP/IP系列協(xié)議歸類(lèi)到四個(gè)抽象層中
應(yīng)用層:TFTP,HTTP箱舞,SNMP遍坟,F(xiàn)TP,SMTP晴股,DNS愿伴,Telnet 等等
傳輸層:TCP,UDP
網(wǎng)絡(luò)層:IP电湘,ICMP隔节,OSPF,EIGRP寂呛,IGMP
數(shù)據(jù)鏈路層:SLIP怎诫,CSLIP,PPP贷痪,MTU
每一抽象層建立在低一層提供的服務(wù)上幻妓,并且為高一層提供服務(wù),看起來(lái)大概是這樣子的
估計(jì)有興趣打開(kāi)此文的同學(xué)都對(duì)此有一定了解了劫拢,加上我也是一知半解肉津,所以就不詳細(xì)解釋?zhuān)信d趣同學(xué)可以上網(wǎng)上搜一下資料
維基百科
百度百科
在TCP/IP協(xié)議中兩個(gè)因特網(wǎng)主機(jī)通過(guò)兩個(gè)路由器和對(duì)應(yīng)的層連接。各主機(jī)上的應(yīng)用通過(guò)一些數(shù)據(jù)通道相互執(zhí)行讀取操作
socket
我們知道兩個(gè)進(jìn)程如果需要進(jìn)行通訊最基本的一個(gè)前提能能夠唯一的標(biāo)示一個(gè)進(jìn)程舱沧,在本地進(jìn)程通訊中我們可以使用PID來(lái)唯一標(biāo)示一個(gè)進(jìn)程妹沙,但PID只在本地唯一,網(wǎng)絡(luò)中的兩個(gè)進(jìn)程PID沖突幾率很大熟吏,這時(shí)候我們需要另辟它徑了距糖,我們知道IP層的ip地址可以唯一標(biāo)示主機(jī),而TCP層協(xié)議和端口號(hào)可以唯一標(biāo)示主機(jī)的一個(gè)進(jìn)程分俯,這樣我們可以利用ip地址+協(xié)議+端口號(hào)唯一標(biāo)示網(wǎng)絡(luò)中的一個(gè)進(jìn)程肾筐。
能夠唯一標(biāo)示網(wǎng)絡(luò)中的進(jìn)程后,它們就可以利用socket進(jìn)行通信了缸剪,什么是socket呢吗铐?我們經(jīng)常把socket翻譯為套接字,socket是在應(yīng)用層和傳輸層之間的一個(gè)抽象層杏节,它把TCP/IP層復(fù)雜的操作抽象為幾個(gè)簡(jiǎn)單的接口供應(yīng)用層調(diào)用已實(shí)現(xiàn)進(jìn)程在網(wǎng)絡(luò)中通信唬渗。
socket起源于UNIX典阵,在Unix一切皆文件哲學(xué)的思想下,socket是一種"打開(kāi)—讀/寫(xiě)—關(guān)閉"模式的實(shí)現(xiàn)镊逝,服務(wù)器和客戶(hù)端各自維護(hù)一個(gè)"文件"壮啊,在建立連接打開(kāi)后,可以向自己文件寫(xiě)入內(nèi)容供對(duì)方讀取或者讀取對(duì)方內(nèi)容撑蒜,通訊結(jié)束時(shí)關(guān)閉文件歹啼。
socket通信流程
socket是"打開(kāi)—讀/寫(xiě)—關(guān)閉"模式的實(shí)現(xiàn),以使用TCP協(xié)議通訊的socket為例座菠,其交互流程大概是這樣子的
服務(wù)器根據(jù)地址類(lèi)型(ipv4,ipv6)狸眼、socket類(lèi)型、協(xié)議創(chuàng)建socket
服務(wù)器為socket綁定ip地址和端口號(hào)
服務(wù)器socket監(jiān)聽(tīng)端口號(hào)請(qǐng)求浴滴,隨時(shí)準(zhǔn)備接收客戶(hù)端發(fā)來(lái)的連接拓萌,這時(shí)候服務(wù)器的socket并沒(méi)有被打開(kāi)
客戶(hù)端創(chuàng)建socket
客戶(hù)端打開(kāi)socket,根據(jù)服務(wù)器ip地址和端口號(hào)試圖連接服務(wù)器socket
服務(wù)器socket接收到客戶(hù)端socket請(qǐng)求升略,被動(dòng)打開(kāi)微王,開(kāi)始接收客戶(hù)端請(qǐng)求,直到客戶(hù)端返回連接信息品嚣。這時(shí)候socket進(jìn)入阻塞狀態(tài)炕倘,所謂阻塞即accept()方法一直到客戶(hù)端返回連接信息后才返回,開(kāi)始接收下一個(gè)客戶(hù)端諒解請(qǐng)求
客戶(hù)端連接成功腰根,向服務(wù)器發(fā)送連接狀態(tài)信息
服務(wù)器accept方法返回激才,連接成功
客戶(hù)端向socket寫(xiě)入信息
服務(wù)器讀取信息
客戶(hù)端關(guān)閉
服務(wù)器端關(guān)閉
三次握手
在TCP/IP協(xié)議中,TCP協(xié)議通過(guò)三次握手建立一個(gè)可靠的連接
第一次握手:客戶(hù)端嘗試連接服務(wù)器额嘿,向服務(wù)器發(fā)送syn包(同步序列編號(hào)Synchronize Sequence Numbers),syn=j劣挫,客戶(hù)端進(jìn)入SYN_SEND狀態(tài)等待服務(wù)器確認(rèn)
第二次握手:服務(wù)器接收客戶(hù)端syn包并確認(rèn)(ack=j+1)册养,同時(shí)向客戶(hù)端發(fā)送一個(gè)SYN包(syn=k),即SYN+ACK包压固,此時(shí)服務(wù)器進(jìn)入SYN_RECV狀態(tài)
第三次握手:第三次握手:客戶(hù)端收到服務(wù)器的SYN+ACK包球拦,向服務(wù)器發(fā)送確認(rèn)包ACK(ack=k+1),此包發(fā)送完畢帐我,客戶(hù)端和服務(wù)器進(jìn)入ESTABLISHED狀態(tài)坎炼,完成三次握手
定睛一看,服務(wù)器socket與客戶(hù)端socket建立連接的部分其實(shí)就是大名鼎鼎的三次握手
socket編程API
前面提到socket是"打開(kāi)—讀/寫(xiě)—關(guān)閉"模式的實(shí)現(xiàn)拦键,簡(jiǎn)單了解一下socket提供了哪些API供應(yīng)用程序使用谣光,還是以TCP協(xié)議為例,看看Unix下的socket API芬为,其它語(yǔ)言都很類(lèi)似(PHP甚至名字都幾乎一樣)萄金,這里我就簡(jiǎn)單解釋一下方法作用和參數(shù)蟀悦,具體使用有興趣同學(xué)可以看看博客參考中的鏈接或者上網(wǎng)搜索
int socket(int domain, int type, int protocol);
根據(jù)指定的地址族、數(shù)據(jù)類(lèi)型和協(xié)議來(lái)分配一個(gè)socket的描述字及其所用的資源氧敢。
domain:協(xié)議族日戈,常用的有AF_INET、AF_INET6孙乖、AF_LOCAL浙炼、AF_ROUTE其中AF_INET代表使用ipv4地址
type:socket類(lèi)型,常用的socket類(lèi)型有唯袄,SOCK_STREAM弯屈、SOCK_DGRAM、SOCK_RAW越妈、SOCK_PACKET季俩、SOCK_SEQPACKET等
protocol:協(xié)議。常用的協(xié)議有梅掠,IPPROTO_TCP酌住、IPPTOTO_UDP、IPPROTO_SCTP阎抒、IPPROTO_TIPC等
int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
把一個(gè)地址族中的特定地址賦給socket
sockfd:socket描述字酪我,也就是socket引用
addr:要綁定給sockfd的協(xié)議地址
addrlen:地址的長(zhǎng)度
通常服務(wù)器在啟動(dòng)的時(shí)候都會(huì)綁定一個(gè)眾所周知的地址(如ip地址+端口號(hào)),用于提供服務(wù)且叁,客戶(hù)就可以通過(guò)它來(lái)接連服務(wù)器都哭;而客戶(hù)端就不用指定,有系統(tǒng)自動(dòng)分配一個(gè)端口號(hào)和自身的ip地址組合逞带。這就是為什么通常服務(wù)器端在listen之前會(huì)調(diào)用bind()欺矫,而客戶(hù)端就不會(huì)調(diào)用,而是在connect()時(shí)由系統(tǒng)隨機(jī)生成一個(gè)展氓。
int listen(int sockfd, int backlog);
監(jiān)聽(tīng)socket
sockfd:要監(jiān)聽(tīng)的socket描述字
backlog:相應(yīng)socket可以排隊(duì)的最大連接個(gè)數(shù)
int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
連接某個(gè)socket
sockfd:客戶(hù)端的socket描述字
addr:服務(wù)器的socket地址
addrlen:socket地址的長(zhǎng)度
int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
TCP服務(wù)器監(jiān)聽(tīng)到客戶(hù)端請(qǐng)求之后穆趴,調(diào)用accept()函數(shù)取接收請(qǐng)求
sockfd:服務(wù)器的socket描述字
addr:客戶(hù)端的socket地址
addrlen:socket地址的長(zhǎng)度
ssize_t read(int fd, void *buf, size_t count);
讀取socket內(nèi)容
fd:socket描述字
buf:緩沖區(qū)
count:緩沖區(qū)長(zhǎng)度
ssize_t write(int fd, const void *buf, size_t count);
向socket寫(xiě)入內(nèi)容,其實(shí)就是發(fā)送內(nèi)容
fd:socket描述字
buf:緩沖區(qū)
count:緩沖區(qū)長(zhǎng)度
int close(int fd);
socket標(biāo)記為以關(guān)閉 遇汞,使相應(yīng)socket描述字的引用計(jì)數(shù)-1未妹,當(dāng)引用計(jì)數(shù)為0的時(shí)候,觸發(fā)TCP客戶(hù)端向服務(wù)器發(fā)送終止連接請(qǐng)求空入。