網絡編程:TCP、UDP及Socket

TCP/IP協議

IP、TCP、UDP 都是TCP/IP協議的一部分。而Socket是應用層與TCP/IP協議通信抽象出來的接口。


TCP/IP.jpg

TCP/IP(Transmission Control Protocol/Internet Protocol)即傳輸控制協議/網間協議,是一個工業標準的協議集,它是為廣域(WANs)設計的。包括運輸層、網絡層、鏈路層。

TCP與UDP的區別

TCP

  1. TCP面向連接(三次握手);UDP是無連接的,即發送數據之前不需要建立連接。
  2. TCP傳輸可靠而UDP不可靠。也就是說,通過TCP連接傳送的數據,無差錯,不丟失,不重復,且按序到達;而UDP則不保證,可能丟包,也不按順序到達。因此UDP更快,效率高,TCP相反。
  3. TCP面向字節流,實際上是TCP把數據看成一連串無結構的字節流。UDP是面向報文的。
  4. 通信方式:每一條TCP連接只能是點到點的;UDP支持一對一,一對多,多對一和多對多的交互通信。
  5. 基于以上這些區別,因此使用場景不同。TCP用在瀏覽器(Http)、文件傳輸(FTP)、接發郵件(SMTP,POP)、遠程登錄(telnet、ssh)。UDP用在視頻通話、語音通話等,適用于多播和廣播的應用場景。

總之,TCP傳播數據準確但速度較慢,用在文件傳輸等對準確性要求較高的地方。UDP相反,用在視頻流等大流量對速度要求較高的地方。

TCP和UDP的Socket實現(JAVA)

Java為Socket編程封裝了幾個重要的類。其中Socket和ServerSocket用于TCP通信。DatagramSocket和DatagramPacket用于UDP通信。


socket.png

基于TCP的Socket編程

Server端
服務器端首先實例化ServerSocket對象,然后為其綁定一個本機地址,并開始監聽。一直阻塞狀態下等待客戶端請求,當獲得客戶端連接請求后,返回一個socket對象。然后用這個socket接收一條消息,并發送一條消息。代碼如下:

package com.chenxuri.java.network;

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * Created by chenxuri on 2017/12/21.
 */
public class SocketTcpServer {
    public static void main(String[] args) {
        try {
            ServerSocket serverSocket = new ServerSocket(8888);
            System.out.println("Connecting to client ...");

            /*
            接收客戶端數據
             */
            Socket socket = serverSocket.accept();
            InputStream inputStream = socket.getInputStream();
            BufferedReader socketIn = new BufferedReader(new InputStreamReader(inputStream));
            String temp;
            while ((temp = socketIn.readLine()) != null) {
                System.out.println(temp);
            }

            /*
            發送數據
             */
            OutputStream outputStream = socket.getOutputStream();
            PrintWriter socketOut = new PrintWriter(outputStream);
            socketOut.print("Hi,I have received your message!");
            socketOut.flush();

            socketOut.close();
            socketIn.close();
            socket.close();
            serverSocket.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

Client端
客戶端首先實例化一個socket對象,用這個對象連接服務器端。連接成功后,發送一條消息,然后等待接收一條消息。代碼如下:

package com.chenxuri.java.network;

import java.io.*;
import java.net.InetSocketAddress;
import java.net.Socket;


/**
 * Created by chenxuri on 2017/12/21.
 */
public class SocketTcpClient {
    public static void main(String[] args) {
        try {
            Socket socket = new Socket();
            /*
            設置地址和連接超時時間,有的地方說不設置連接時間可能會造成無限期阻塞。
            其實不會,因為操作系統底層有超時限制,windows是20秒。
            但仍建議設置一個較短的時間,尤其是需要頻繁連接的時候。
             */
            socket.connect(new InetSocketAddress("127.0.0.1",8888), 5*1000);
            //設置讀取超時時間,該時間若不設置,服務器等出現問題時可能會一直處于阻塞狀態。
            socket.setSoTimeout(10*1000);

            /*
            往服務端發送數據
             */
            OutputStream outputStream  = socket.getOutputStream();
            PrintWriter socketOut = new PrintWriter(outputStream);
            String str = "Hello,I come from client!";
            socketOut.write(str);
            socketOut.flush();
            socket.shutdownOutput();  //半關閉,告訴服務端發送完畢,可以接受輸入過來的數據

             /*
            接收回應數據
             */
            InputStream inputStream = socket.getInputStream();
            BufferedReader socketIn = new BufferedReader(new InputStreamReader(inputStream));
            String temp;
            while ((temp = socketIn.readLine()) != null) {
                System.out.println(temp);
            }

            socketIn.close();
            socketOut.close();
            socket.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

基于UDP的Socket編程

Server端
服務器端首先實例化DatagramSocket對象,然后為其綁定一個端口,并開始監聽。一直阻塞狀態下等待從客戶端接收數據報。然后從數據報中獲取數據報的源地址,然后用這個源地址作為目的地址打包一個數據報,然后發送出去。代碼如下:

package com.chenxuri.java.network;

import java.io.*;
import java.net.*;

/**
 * Created by chenxuri on 2017/12/21.
 */
public class SocketUdpServer {
    public static void main(String[] args) {
        try {
            DatagramSocket socket = new DatagramSocket(8888);

            /*
            接收客戶端的數據
             */
            byte[] bytes = new byte[1024];
            DatagramPacket packet = new DatagramPacket(bytes,bytes.length);
            socket.receive(packet);
            String receiveStr = new String(bytes);
            System.out.println("From client: " + receiveStr);

            /*
            發送回應數據
             */
            int port = packet.getPort();
            InetAddress addr = packet.getAddress();
            String sendStr = "Hello! I'm Server";
            byte[] sendBuf = sendStr.getBytes();
            DatagramPacket sendPacket = new DatagramPacket(sendBuf , sendBuf.length , addr , port );
            socket.send(sendPacket);
            socket.close();

        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

Client端
客戶端首先實例化一個DatagramSocket對象。利用服務器地址和端口號作為目的地址打包一個數據報,并發送。然后等待從服務器回復的數據報。代碼如下:

package com.chenxuri.java.network;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.InetSocketAddress;

/**
 * Created by chenxuri on 2017/12/21.
 */
public class SocketUdpClient {
    public static void main(String[] args) {
        try {
            DatagramSocket socket = new DatagramSocket();

            /*
            發送數據
             */
            String sendStr = "I'm client, this is the message for server.";
            byte[] bytes = sendStr.getBytes();
            DatagramPacket packet = new DatagramPacket(bytes,bytes.length);
            packet.setSocketAddress(new InetSocketAddress("127.0.0.1",8888));
            socket.send(packet);

            /*
            接收返回數據
             */
            byte[] backbuf = new byte[1024];
            DatagramPacket backPacket = new DatagramPacket(backbuf, backbuf.length);
            socket.receive(backPacket);
            System.out.println("From server: "+ new String(backbuf));

            socket.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

以上代碼都經過親自測試,可以運行。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,836評論 6 540
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,275評論 3 428
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,904評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,633評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,368評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,736評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,740評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,919評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,481評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,235評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,427評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,968評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,656評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,055評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,348評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,160評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,380評論 2 379

推薦閱讀更多精彩內容

  • 個人認為,Goodboy1881先生的TCP /IP 協議詳解學習博客系列博客是一部非常精彩的學習筆記,這雖然只是...
    貳零壹柒_fc10閱讀 5,084評論 0 8
  • 1.這篇文章不是本人原創的,只是個人為了對這部分知識做一個整理和系統的輸出而編輯成的,在此鄭重地向本文所引用文章的...
    SOMCENT閱讀 13,115評論 6 174
  • 計算機網絡七層模型中,傳輸層有兩個重要的協議:(1)用戶數據報協議UDP (User Datagram Proto...
    Q南南南Q閱讀 1,734評論 0 3
  • 轉。。。。。。。。 SOCKET,TCP/UDP,HTTP,FTP (一)TCP/UDP,SOCKET,HTTP,...
    zeqinjie閱讀 3,308評論 1 53
  • 一個偶然的機會,接觸了果語零食機,一開始自己綜合考慮,加個人愛好買了倆,一個冰棒機,一個原汁機,抱著試試看的心態,...
    雁南飛fly閱讀 473評論 7 7