本章將討論如何優(yōu)雅地?cái)嚅_(kāi)相互連接的套接字。之前用的方法不夠優(yōu)雅是因?yàn)椋覀兪钦{(diào)用close或closesocket函數(shù)單方面斷開(kāi)連接的。
創(chuàng)新互聯(lián)是一家專(zhuān)業(yè)提供太白企業(yè)網(wǎng)站建設(shè),專(zhuān)注與網(wǎng)站設(shè)計(jì)、成都做網(wǎng)站、HTML5、小程序制作等業(yè)務(wù)。10年已為太白眾多企業(yè)、政府機(jī)構(gòu)等服務(wù)。創(chuàng)新互聯(lián)專(zhuān)業(yè)網(wǎng)站設(shè)計(jì)公司優(yōu)惠進(jìn)行中?;赥CP的半關(guān)閉TCP中的斷開(kāi)連接過(guò)程比建立連接過(guò)程更重要,因?yàn)檫B接過(guò)程中一般不會(huì)出現(xiàn)大的變數(shù),但斷開(kāi)過(guò)程有可能發(fā)生預(yù)想不到的情況,因此應(yīng)準(zhǔn)確掌控。只有掌握了下面要講解的半關(guān)閉(Half-close),才能明確斷開(kāi)過(guò)程。
單方面斷開(kāi)連接帶來(lái)的問(wèn)題
Linux的close函數(shù)和Windows的closesocket函數(shù)意味著完全斷開(kāi)連接。完全斷開(kāi)不僅指無(wú)法傳輸數(shù)據(jù),而且也不能接收數(shù)據(jù)。因此,在某些情況下,通信一方調(diào)用close或closesocket函數(shù)斷開(kāi)連接就顯得不太優(yōu)雅,如圖7-1所示。
圖7-1描述的是2臺(tái)主機(jī)正在進(jìn)行雙向通信。主機(jī)A發(fā)送完最后的數(shù)據(jù)后,調(diào)用close函數(shù)斷開(kāi)了連接,之后主機(jī)A無(wú)法再接收主機(jī)B傳輸?shù)臄?shù)據(jù)。實(shí)際上,是完全無(wú)法調(diào)用與接收數(shù)據(jù)相關(guān)的函數(shù)。最終,由主機(jī)B傳輸?shù)?、主機(jī)A必須接收的數(shù)據(jù)也銷(xiāo)毀了。
為了解決這類(lèi)問(wèn)題,“只關(guān)閉一部分?jǐn)?shù)據(jù)交換中使用的流”(Half-close)的方法應(yīng)運(yùn)而生。斷開(kāi)一部分連接是指,可以傳輸數(shù)據(jù)但無(wú)法接收,或可以接收數(shù)據(jù)但無(wú)法傳輸。顧名思義就是只關(guān)閉流的一半。
套接字和流(Stream)
兩臺(tái)主機(jī)通過(guò)套接字建立連接后進(jìn)入可交換數(shù)據(jù)的狀態(tài),又稱“流形成的狀態(tài)”。也就是把建立套接字后可交換數(shù)據(jù)的狀態(tài)看作一種流。
此處的流可以比作水流。水朝著一個(gè)方向流動(dòng),同樣,在套接字的流中,數(shù)據(jù)也只能向一個(gè)方向移動(dòng)。因此,為了進(jìn)行雙向通信,需要如圖7-2所示的2個(gè)流。
一旦兩臺(tái)主機(jī)間建立了套接字連接,每個(gè)主機(jī)就會(huì)擁有單獨(dú)的輸入流和輸出流。當(dāng)然,其中一個(gè)主機(jī)的輸入流與另一主機(jī)的輸出流相連,而輸出流則與另一主機(jī)的輸入流相連。另外,本章討論的“優(yōu)雅地?cái)嚅_(kāi)連接方式”只斷開(kāi)其中1個(gè)流,而非同時(shí)斷開(kāi)兩個(gè)流。Linux的close和Windows 的closesocket函數(shù)將同時(shí)斷開(kāi)這兩個(gè)流,因此與"優(yōu)雅"二字還有一段距離。
針對(duì)優(yōu)雅斷開(kāi)的shutdown函數(shù)
接下來(lái)介紹用于半關(guān)閉的函數(shù)。下面這個(gè)shutdown函數(shù)就用來(lái)關(guān)閉其中1個(gè)流。
#includeint shutdown(int sock, int howto);
// 成功時(shí)返回0,失敗時(shí)返回-1。
// sock 需要斷開(kāi)的套接字文件描述符。
// howto 傳遞斷開(kāi)方式信息。
調(diào)用上述函數(shù)時(shí),第二個(gè)參數(shù)決定斷開(kāi)連接的方式,其可能值如下所示。
若向shutdown的第二個(gè)參數(shù)傳遞SHUT_RD,則斷開(kāi)輸入流,套接字無(wú)法接收數(shù)據(jù)。即使輸入緩沖收到數(shù)據(jù)也會(huì)抹去,而且無(wú)法調(diào)用輸入相關(guān)函數(shù)。如果向shutdown函數(shù)的第二個(gè)參數(shù)傳遞SHUT_WR,則中斷輸出流,也就無(wú)法傳輸數(shù)據(jù)。但如果輸出緩沖還留有未傳輸?shù)臄?shù)據(jù),則將傳遞至目標(biāo)主機(jī)。最后,若傳入SHUT_RDWR,則同時(shí)中斷I/O流。這相當(dāng)于分2次調(diào)用shutdown,其中一次以SHUT_RD為參數(shù),另一次以SHUT_WR為參數(shù)。
為何需要半關(guān)閉
相信各位已對(duì)"關(guān)閉套接字的一半連接"有了充分的認(rèn)識(shí),但還有一些疑惑。
“究竟為什么需要半關(guān)閉?是否只要留出足夠長(zhǎng)的連接時(shí)間,保證完成數(shù)據(jù)交換即可?只要不急于斷開(kāi)連接,好像也沒(méi)必要使用半關(guān)閉。"
這句話也不完全是錯(cuò)的。如果保持足夠的時(shí)間間隔,完成數(shù)據(jù)交換后再斷開(kāi)連接,這時(shí)就沒(méi)必要使用半關(guān)閉。但要考慮如下情況:
"一旦客戶端連接到服務(wù)器端,服務(wù)器端將約定的文件傳給客戶端,客戶端收到后發(fā)送字符串‘Thank you’給服務(wù)器端?!?/p>
此處字符串"Thank you"的傳遞實(shí)際是多余的,這只是用來(lái)模擬客戶端斷開(kāi)連接前還有數(shù)據(jù)需要傳遞的情況。此時(shí)程序?qū)崿F(xiàn)的難度并不小,因?yàn)閭鬏斘募姆?wù)器端只需連續(xù)傳輸文件數(shù)據(jù)即可,而客戶端則無(wú)法知道需要接收數(shù)據(jù)到何時(shí)??蛻舳艘矝](méi)辦法無(wú)休止地調(diào)用輸入函數(shù),因?yàn)檫@有可能導(dǎo)致程序阻塞(調(diào)用的函數(shù)未返回)。
“是否可以讓服務(wù)器端和客戶端約定一個(gè)代表文件尾的字符?”
這種方式也有問(wèn)題,因?yàn)檫@意味著文件中不能有與約定字符相同的內(nèi)容。為解決該問(wèn)題,服務(wù)器端應(yīng)最后向客戶端傳遞EOF表示文件傳輸結(jié)束??蛻舳送ㄟ^(guò)函數(shù)返回值接收EOF,這樣可以避免與文件內(nèi)容沖突。剩下最后一個(gè)問(wèn)題:服務(wù)器如何傳遞EOF?
“斷開(kāi)輸出流時(shí)向?qū)Ψ街鳈C(jī)傳輸EOF。”
當(dāng)然,調(diào)用close函數(shù)的同時(shí)關(guān)閉I/O流,這樣也會(huì)向?qū)Ψ桨l(fā)送EOF。但此時(shí)無(wú)法再接收對(duì)方傳輸?shù)臄?shù)據(jù)。換言之,若調(diào)用close函數(shù)關(guān)閉流,就無(wú)法接收客戶端最后發(fā)送的字符串"Thank you"。這時(shí)需要調(diào)用shutdown函數(shù),只關(guān)閉服務(wù)器的輸出流(半關(guān)閉)。這樣既可以發(fā)送EOF,同時(shí)又保留了輸入流,可以接收對(duì)方數(shù)據(jù)。
基于Windows的實(shí)現(xiàn)跳過(guò)
你是否還在尋找穩(wěn)定的海外服務(wù)器提供商?創(chuàng)新互聯(lián)www.cdcxhl.cn海外機(jī)房具備T級(jí)流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確流量調(diào)度確保服務(wù)器高可用性,企業(yè)級(jí)服務(wù)器適合批量采購(gòu),新人活動(dòng)首月15元起,快前往官網(wǎng)查看詳情吧
標(biāo)題名稱:07.優(yōu)雅地?cái)嚅_(kāi)套接字連接-創(chuàng)新互聯(lián)
文章鏈接:http://jinyejixie.com/article14/ccegde.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供ChatGPT、域名注冊(cè)、全網(wǎng)營(yíng)銷(xiāo)推廣、定制網(wǎng)站、App設(shè)計(jì)、品牌網(wǎng)站建設(shè)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容