建立TCP需要三次握手才能建立婴程,而斷開連接則需要四次握手庄敛。整個過程如下圖所示:
先來看看如何建立連接的伟叛。
【更新于2017.01.04 】該部分內(nèi)容配圖有誤略就,請大家見諒兄墅,正確的配圖如下踢星,錯誤配圖也不刪了,大家可以比較下察迟,對比理解效果更好斩狱。這么久才來更新,抱歉T俊所踊!
錯誤配圖如下:
首先Client端發(fā)送連接請求報文,Server段接受連接后回復(fù)ACK報文概荷,并為這次連接分配資源秕岛。Client端接收到ACK報文后也向Server段發(fā)生ACK報文,并分配資源误证,這樣TCP連接就建立了继薛。
那如何斷開連接呢?簡單的過程如下:
【注意】中斷連接端可以是Client端愈捅,也可以是Server端遏考。
假設(shè)Client端發(fā)起中斷連接請求,也就是發(fā)送FIN報文蓝谨。Server端接到FIN報文后灌具,意思是說"我Client端沒有數(shù)據(jù)要發(fā)給你了",但是如果你還有數(shù)據(jù)沒有發(fā)送完成譬巫,則不必急著關(guān)閉Socket咖楣,可以繼續(xù)發(fā)送數(shù)據(jù)。所以你先發(fā)送ACK芦昔,"告訴Client端诱贿,你的請求我收到了,但是我還沒準(zhǔn)備好咕缎,請繼續(xù)你等我的消息"珠十。這個時候Client端就進入FIN_WAIT狀態(tài),繼續(xù)等待Server端的FIN報文凭豪。當(dāng)Server端確定數(shù)據(jù)已發(fā)送完成宵睦,則向Client端發(fā)送FIN報文,"告訴Client端墅诡,好了,我這邊數(shù)據(jù)發(fā)完了,準(zhǔn)備好關(guān)閉連接了"末早。Client端收到FIN報文后烟馅,"就知道可以關(guān)閉連接了,但是他還是不相信網(wǎng)絡(luò)然磷,怕Server端不知道要關(guān)閉郑趁,所以發(fā)送ACK后進入TIME_WAIT狀態(tài),如果Server端沒有收到ACK則可以重傳姿搜」讶螅“,Server端收到ACK后舅柜,"就知道可以斷開連接了"梭纹。Client端等待了2MSL后依然沒有收到回復(fù),則證明Server端已正常關(guān)閉致份,那好变抽,我Client端也可以關(guān)閉連接了。Ok氮块,TCP連接就這樣關(guān)閉了绍载!
整個過程Client端所經(jīng)歷的狀態(tài)如下:
而Server端所經(jīng)歷的過程如下:轉(zhuǎn)載請注明:blog.csdn.net/whuslei
【注意】 在TIME_WAIT狀態(tài)中,如果TCP client端最后一次發(fā)送的ACK丟失了滔蝉,它將重新發(fā)送击儡。TIME_WAIT狀態(tài)中所需要的時間是依賴于實現(xiàn)方法的。典型的值為30秒蝠引、1分鐘和2分鐘阳谍。等待之后連接正式關(guān)閉,并且所有的資源(包括端口號)都被釋放立肘。
【問題1】為什么連接的時候是三次握手边坤,關(guān)閉的時候卻是四次握手?答:因為當(dāng)Server端收到Client端的SYN連接請求報文后谅年,可以直接發(fā)送SYN+ACK報文茧痒。其中ACK報文是用來應(yīng)答的,SYN報文是用來同步的融蹂。但是關(guān)閉連接時旺订,當(dāng)Server端收到FIN報文時,很可能并不會立即關(guān)閉SOCKET超燃,所以只能先回復(fù)一個ACK報文区拳,告訴Client端,"你發(fā)的FIN報文我收到了"意乓。只有等到我Server端所有的報文都發(fā)送完了樱调,我才能發(fā)送FIN報文,因此不能一起發(fā)送。故需要四步握手笆凌。
【問題2】為什么TIME_WAIT狀態(tài)需要經(jīng)過2MSL(最大報文段生存時間)才能返回到CLOSE狀態(tài)圣猎?
答:雖然按道理,四個報文都發(fā)送完畢乞而,我們可以直接進入CLOSE狀態(tài)了送悔,但是我們必須假象網(wǎng)絡(luò)是不可靠的,有可以最后一個ACK丟失爪模。所以TIME_WAIT狀態(tài)就是用來重發(fā)可能丟失的ACK報文欠啤。