Java TCP/UDP socket 編程流程

最近正好學習了一點用java socket編程的東西硕旗。感覺整體的流程雖然不是很繁瑣乡范,但是也值得好好總結一下困后。

Socket

Socket可以說是一種針對網絡的抽象豆胸,應用通過它可以來針對網絡讀寫數(shù)據。就像通過一個文件的file handler就可以都寫數(shù)據到存儲設備上一樣赤套。根據TCP協(xié)議和UDP協(xié)議的不同飘痛,在網絡編程方面就有面向兩個協(xié)議的不同socket,一個是面向字節(jié)流的一個是面向報文的于毙。

對socket的本身組成倒是比較好理解敦冬。既然是應用通過socket通信辅搬,肯定就有一個服務器端和一個客戶端唯沮。所以它必然就包含有一個對應的IP地址脖旱。另外,在這個地址上server要提供一系列的服務介蛉,于是就需要有一系列對應的窗口來提供服務萌庆。所以就有一個對應的端口號(Port)。端口號是一個16位的二進制數(shù)字币旧,那么范圍就是從(0-65535)践险。IP地址加端口號基本上就構成了socket。下面這幅圖可以描繪出socket和整個TCP/IP之間的關系:

TCP

TCP主要是面向連接的協(xié)議吹菱,它包含有建立和拆除連接巍虫,保證數(shù)據流的順序和正確性等功能。每次對TCP中間的數(shù)據操作相當于對一個數(shù)據流進行訪問鳍刷。它最典型的特征就是那三次握手的建立連接過程占遥。TCP的連接建立和撤銷過程如下圖:

Server端

Server端所要做的事情主要是建立一個通信的端點,然后等待客戶端發(fā)送的請求输瓜。典型的處理步驟如下:

1. 構建一個ServerSocket實例瓦胎,指定本地的端口。這個socket就是用來監(jiān)聽指定端口的連接請求的尤揣。

2.重復如下幾個步驟:

a. 調用socket的accept()方法來獲得下面客戶端的連接請求搔啊。通過accept()方法返回的socket實例,建立了一個和客戶端的新連接北戏。

b.通過這個返回的socket實例獲取InputStream和OutputStream,可以通過這兩個stream來分別讀和寫數(shù)據负芋。

c.結束的時候調用socket實例的close()方法關閉socket連接。

這個流程的典型示例代碼如下:

Java代碼

//1.?構造ServerSocket實例嗜愈,指定服務端口示罗。

ServerSocket?servSock?=newServerSocket(servPort);

while(true)

{

//?2.調用accept方法,建立和客戶端的連接

Socket?clntSock?=?servSock.accept();

SocketAddress?clientAddress?=

clntSock.getRemoteSocketAddress();

System.out.println("Handling?client?at?"+?clientAddress);

//?3.?獲取連接的InputStream,OutputStream來進行數(shù)據讀寫

InputStream?in?=?clntSock.getInputStream();

OutputStream?out?=?clntSock.getOutputStream();

while((recvMsgSize?=?in.read(receiveBuf))?!=?-1)

{

out.write(receiveBuf,0,?recvMsgSize);

}

//?4.操作結束芝硬,關閉socket.

clntSock.close();

}

Client端

客戶端的請求過程稍微有點不一樣:

1.構建Socket實例蚜点,通過指定的遠程服務器地址和端口來建立連接。

2.通過Socket實例包含的InputStream和OutputStream來進行數(shù)據的讀寫拌阴。

3.操作結束后調用socket實例的close方法绍绘,關閉。

示例代碼如下迟赃;

Java代碼

//?1.根據指定的server地址和端口陪拘,建立socket連接。

Socket?socket?=newSocket(server,?servPort);

//?2.?根據socket實例獲取InputStream,?OutputStream進行數(shù)據讀寫纤壁。

InputStream?in?=?socket.getInputStream();

OutputStream?out?=?socket.getOutputStream();

out.write(data);

//3.操作結束左刽,關閉socket.

socket.close();

UDP

UDP和TCP有兩個典型的區(qū)別,一個就是它不需要建立連接酌媒,另外就是它在每次收發(fā)的報文都保留了消息的邊界欠痴。

server端

因為UDP協(xié)議不需要建立連接迄靠,它的過程如下:

1. 構造DatagramSocket實例,指定本地端口喇辽。

2. 通過DatagramSocket實例的receive方法接收DatagramPacket.DatagramPacket中間就包含了通信的內容掌挚。

3. 通過DatagramSocket的send和receive方法來收和發(fā)DatagramPacket.

典型的交互流程代碼如下:

Java代碼

//?1.?構建DatagramSocket實例,指定本地端口菩咨。

DatagramSocket?socket?=newDatagramSocket(servPort);

//?2.?構建需要收發(fā)的DatagramPacket報文

DatagramPacket?packet?=newDatagramPacket(newbyte[ECHOMAX],?ECHOMAX);

while(true)

{

//?3.?收報文

socket.receive(packet);

System.out.println("Handling?client?at?"+?packet.getAddress().getHostAddress()

+"?on?port?"+?packet.getPort());

//?4.?發(fā)報文

socket.send(packet);

packet.setLength(ECHOMAX);

}

client端

UDP客戶端的步驟也比較簡單吠式,主要包括下面3步:

1. 構造DatagramSocket實例。

2.通過DatagramSocket實例的send和receive方法發(fā)送DatagramPacket報文抽米。

3.結束后特占,調用DatagramSocket的close方法關閉。

因為和TCP不同云茸,UDP發(fā)送報文的時候可以在同一個本地端口隨意發(fā)送給不同的服務器摩钙,一般不需要在UDP的DatagramSocket的構造函數(shù)中指定目的服務器的地址。

另外查辩,UDP客戶端還有一個重要的不同就是胖笛,TCP客戶端發(fā)送echo連接消息之后會在調用read方法的時候進入阻塞狀態(tài),而UDP這樣卻不行宜岛。因為UDP中間是可以允許報文丟失的长踊。如果報文丟失了,進程一直在阻塞或者掛起的狀態(tài)萍倡,則進程會永遠沒法往下走了身弊。所以會一般設置一個setSoTimeout方法,指定在多久的時間內沒有收到報文就放棄列敲。也可以通過指定一個數(shù)字阱佛,循環(huán)指定的次數(shù)來讀取報文,讀到就返回戴而,否則就放棄凑术。

一個典型的UDP Client代碼示例如下:

Java代碼

//?1.?構造UDP?DatagramSocket對象

DatagramSocket?socket?=newDatagramSocket();

//?2。指定timeout時間所意,防止進入無限等待狀態(tài)

socket.setSoTimeout(TIMEOUT);

//?3.?構造收發(fā)的報文對象

DatagramPacket?sendPacket?=newDatagramPacket(bytesToSend,

bytesToSend.length,?serverAddress,?servPort);

DatagramPacket?receivePacket?=

newDatagramPacket(newbyte[bytesToSend.length],?bytesToSend.length);

//?4.指定嘗試的次數(shù)

inttries?=0;

booleanreceivedResponse?=false;

do

{

socket.send(sendPacket);

try

{

socket.receive(receivePacket);

if(!receivePacket.getAddress().equals(serverAddress))

{

thrownewIOException("Received?packet?from?an?unknown?source");

}

receivedResponse?=true;

}

catch(InterruptedIOException?e)

{

tries?+=1;

System.out.println("Timed?out,?"+?(MAXTRIES?-?tries)?+"");

}

}while((!receivedResponse)?&&?(tries?<?MAXTRIES));

//?根據是否接收到報文進行反饋

if(receivedResponse)

{

System.out.println("Received:?"+newString(receivePacket.getData()));

}

else

{

System.out.println("No?response?--?giving?up.");

}

//?5.?關閉socket

socket.close();

總結

TCP的server和client之間通信就好比兩個人打電話淮逊,需要互相知道對方的電話號碼,然后開始對話扶踊。所以在兩者的連接過程中間需要指定端口和地址泄鹏。

UDP的server和client之間的通信就像兩個人互相發(fā)信。我只需要知道對方的地址秧耗,然后就發(fā)信過去备籽。對方是否收到我不知道,也不需要專門對口令似的來建立連接分井。

這些示例其實只是一個最簡單的车猬,單線程霉猛,也只能一次處理單個請求的情況。在實際應用中一般會應用到多線程和一些處理高并發(fā)的策略诈唬,比如基于事件驅動的reactor模式等∷豸铮可以在后續(xù)的文章中深入討論铸磅。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市杭朱,隨后出現(xiàn)的幾起案子阅仔,更是在濱河造成了極大的恐慌,老刑警劉巖弧械,帶你破解...
    沈念sama閱讀 217,185評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件八酒,死亡現(xiàn)場離奇詭異,居然都是意外死亡刃唐,警方通過查閱死者的電腦和手機羞迷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,652評論 3 393
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來画饥,“玉大人衔瓮,你說我怎么就攤上這事《陡剩” “怎么了热鞍?”我有些...
    開封第一講書人閱讀 163,524評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長衔彻。 經常有香客問我薇宠,道長,這世上最難降的妖魔是什么艰额? 我笑而不...
    開封第一講書人閱讀 58,339評論 1 293
  • 正文 為了忘掉前任澄港,我火速辦了婚禮,結果婚禮上柄沮,老公的妹妹穿的比我還像新娘慢睡。我一直安慰自己,他們只是感情好铡溪,可當我...
    茶點故事閱讀 67,387評論 6 391
  • 文/花漫 我一把揭開白布漂辐。 她就那樣靜靜地躺著,像睡著了一般棕硫。 火紅的嫁衣襯著肌膚如雪髓涯。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,287評論 1 301
  • 那天哈扮,我揣著相機與錄音纬纪,去河邊找鬼蚓再。 笑死,一個胖子當著我的面吹牛包各,可吹牛的內容都是我干的摘仅。 我是一名探鬼主播,決...
    沈念sama閱讀 40,130評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼问畅,長吁一口氣:“原來是場噩夢啊……” “哼娃属!你這毒婦竟也來了?” 一聲冷哼從身側響起护姆,我...
    開封第一講書人閱讀 38,985評論 0 275
  • 序言:老撾萬榮一對情侶失蹤矾端,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后卵皂,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體秩铆,經...
    沈念sama閱讀 45,420評論 1 313
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,617評論 3 334
  • 正文 我和宋清朗相戀三年灯变,在試婚紗的時候發(fā)現(xiàn)自己被綠了殴玛。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,779評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡添祸,死狀恐怖族阅,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情膝捞,我是刑警寧澤坦刀,帶...
    沈念sama閱讀 35,477評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站蔬咬,受9級特大地震影響鲤遥,放射性物質發(fā)生泄漏。R本人自食惡果不足惜林艘,卻給世界環(huán)境...
    茶點故事閱讀 41,088評論 3 328
  • 文/蒙蒙 一盖奈、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧狐援,春花似錦钢坦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,716評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至镶殷,卻和暖如春禾酱,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,857評論 1 269
  • 我被黑心中介騙來泰國打工颤陶, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留颗管,地道東北人。 一個月前我還...
    沈念sama閱讀 47,876評論 2 370
  • 正文 我出身青樓滓走,卻偏偏與公主長得像垦江,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子搅方,可洞房花燭夜當晚...
    茶點故事閱讀 44,700評論 2 354

推薦閱讀更多精彩內容