Java Socket實現基於TCP和UDP多線程通信

来源:http://www.cnblogs.com/zhaozihan/archive/2016/11/12/6057118.html
-Advertisement-
Play Games

一.通過Socket實現TCP編程 1.1 TCP編程 TCP協議是面向連接,可靠的,有序的,以位元組流的方式發送數據。基於TCP協議實現網路通信的類有客戶端的Socket類和伺服器端的ServerSocket類。 1.2 伺服器端套路 1.創建ServerSocket對象,綁定監聽埠。 2.通過a ...


一.通過Socket實現TCP編程

1.1 TCP編程

  TCP協議是面向連接,可靠的,有序的,以位元組流的方式發送數據。基於TCP協議實現網路通信的類有客戶端的Socket類和伺服器端的ServerSocket類。

1.2 伺服器端套路

  1.創建ServerSocket對象,綁定監聽埠。

  2.通過accept()方法監聽客戶端請求。

  3.連接建立後,通過輸入流讀取客戶端發送的請求信息。

  4.通過輸出流向客戶端發送響應信息。

  5.關閉響應的資源。

1.3 客戶端套路

  1.創建Socket對象,指明需要連接的伺服器的地址和埠號。

  2.連接建立後,通過輸出流向伺服器發送請求信息。

  3.通過輸入流獲取伺服器響應的信息。

  4.關閉相應資源。

1.4 多線程實現伺服器與多客戶端之間通信步驟

  1.伺服器端創建ServerSocket,迴圈調用accept()等待客戶端連接。

  2. 客戶端創建一個socket並請求和伺服器端連接。

  3.伺服器端接受客戶端請求,創建socket與該客戶建立專線連接。

  4.建立連接的兩個socket在一個單獨的線程上對話。

  5.伺服器端繼續等待新的連接。

1.5 創建處理線程類ServerThread

  這裡選擇實現runnable介面而不是繼承Thread是因為一個類只能繼承一個父類,當我需要繼承其他類的時,父類就就不好處理了。

  

package com.tzzh;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.Socket;

public class ServerThread implements Runnable{

    Socket socket = null;//和本線程相關的Socket
        
    public ServerThread(Socket socket) {
    this.socket = socket;
}

    @Override
    public void run() {
        InputStream is = null;
        InputStreamReader isr = null;
        BufferedReader br = null;
        OutputStream os = null;
        PrintWriter pw = null;
        try {
            //與客戶端建立通信,獲取輸入流,讀取取客戶端提供的信息
            is = socket.getInputStream();
            isr = new InputStreamReader(is,"GBK");
            br = new BufferedReader(isr);
            String data = null;
            while((data=br.readLine()) != null){//迴圈讀取客戶端的信息
                System.out.println("我是伺服器,客戶端提交信息為:"+data);
            }
            socket.shutdownInput();//關閉輸入流
            
            //獲取輸出流,響應客戶端的請求
            os = socket.getOutputStream();
            pw = new PrintWriter(os);
            pw.write("伺服器端響應成功!");
            pw.flush();
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            //關閉資源即相關socket
            try {
                if(pw!=null)
                    pw.close();
                if(os!=null)
                    os.close();
                if(br!=null)
                    br.close();
                if(isr!=null)
                    isr.close();
                if(is!=null)
                    is.close();
                if(socket!=null)
                    socket.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            
        }
        
    }

}

 

1.6 創建伺服器端類

  使用while以達到可以迴圈偵聽不同客戶端的連接請求。因為這是一個死迴圈,所以不用關閉也沒有機會去關閉serverSocket。設置count值,用於記錄伺服器端被連接過的次數並顯示客戶端所在ip值。如果線程處理類是繼承Thread類,那麼創建新線程代碼可以改為ServerThread serverThread = new ServerThread(socket);serverThread.start();

package com.tzzh;

import java.io.IOException;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;

public class Server {
    public static void main(String[] args) {
        try {
            //創建一個伺服器端的Socket,即ServerSocket,綁定需要監聽的埠
            ServerSocket serverSocket = new ServerSocket(8888);
            Socket socket = null;
            //記錄連接過伺服器的客戶端數量
            int count = 0;
            System.out.println("***伺服器即將啟動,等待客戶端的連接***");
            while(true){//迴圈偵聽新的客戶端的連接
                //調用accept()方法偵聽,等待客戶端的連接以獲取Socket實例
                socket = serverSocket.accept();
                //創建新線程
                Thread thread = new Thread(new ServerThread(socket));
                thread.start();

                count++;
                System.out.println("伺服器端被連接過的次數:"+count);
                InetAddress address = socket.getInetAddress();
                System.out.println("當前客戶端的IP為:"+address.getHostAddress());
            }            
            //serverSocket.close();一直迴圈監聽,不用關閉連接
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

 1.7 創建客戶端類

  在後面的關閉資源中,我把輸入輸出相關的流關閉註釋了,是因為對於同一個Socket,關閉socket的時候也會把輸入輸出流關閉,直接關閉socket就行,當然保留也是可以的。

package com.tzzh;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.Socket;
import java.net.UnknownHostException;

public class Client {
    public static void main(String[] args) {
        try {
            //創建客戶端Socket,指定伺服器地址和埠
            Socket socket = new Socket("localhost", 8888);
            //建立連接後,獲取輸出流,向伺服器端發送信息
            OutputStream os = socket.getOutputStream();
            //輸出流包裝為列印流
            PrintWriter pw = new PrintWriter(os);
            //向伺服器端發送信息
            pw.write("用戶名:zzh;密碼:123");//寫入記憶體緩衝區
            pw.flush();//刷新緩存,向伺服器端輸出信息
            socket.shutdownOutput();//關閉輸出流
            
            //獲取輸入流,接收伺服器端響應信息
            InputStream is = socket.getInputStream();
            BufferedReader br = new BufferedReader(new InputStreamReader(is, "GBK"));
            String data = null;
            while((data=br.readLine())!= null){
                System.out.println("我是客戶端,伺服器端提交信息為:"+data);
            }
            
            //關閉其他資源
//            br.close();
//            is.close();
//            pw.close();
//            os.close();
            socket.close();
        
        } catch (UnknownHostException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

 

1.8 先運行伺服器端,在運行客戶端

 

此時在看伺服器控制台:伺服器端一直在迴圈偵聽客戶端的連接

 

 1.9 進行第二個客戶端的連接

  修改相應信息將用戶名zzh改為admin。運行客戶端,打開服務端控制台

  

輸出客戶端的ip都為127.0.0.1,是因為伺服器和客戶端都是本機,在真實的環境中會顯示客戶端的ip地址信息。

 

二. 通過Socket實現UDP編程

2.1 UDP編程

  UDP協議又叫用戶數據報協議,是無連接,不可靠的,無序的。特點是傳輸速度相對要快,UDP協議以數據報作為數據傳輸的載體。當進行數據傳輸時,首先需要將要傳輸的數據定義成數據報(Datagram),在數據報中指明數據所要達到的Socket(主機地址和埠號),然後再將數據報發送出去。相關操作類有:DatagramPacket數據報包,DatagramSocket進行端到端通信的類。

2.2 伺服器端實現套路

  1.創建DatagramSocket,指定埠號。2.創建DatagramPacket。3.接收客戶端發送的數據信息。4.讀取數據。

2.3 客戶端實現套路

  1.定義發送信息,比如發送地址,埠號和內容。2. 創建DatagramPacket,包含將要發送的信息。3.創建DatagramSocket。4.發送數據。

2.4 多線程實現伺服器與多客戶端之間通信步驟

  1.伺服器端創建DatagramSocket的實例socket,迴圈調用receive()方法,此方法在接收到數據報之前會一直阻塞

  2.客戶端創建DatagramSocket,將含有地址,埠號和內容的數據報包發送出去。

  3. 伺服器端收到數據報包packet,通過DatagramSocket和packet與客戶端建立一個線程

  4. 伺服器端繼續等待新的數據報包。

  5. 發送方的DatagramPacket構造方法傳遞四個參數包含數據內容,數據大小,地址和埠號。接收方的DatagramPacket構造方法有兩個參數接收數據和數據大小。

2.5 創建伺服器線程處理類UDPThread

  註意,DatagramSocket的實例socket不能關閉,會出現SocketException。讀取數據用到的new String(packet.getData(), 0, packet.getLength()),參數表示數據報中的位元組數組,位置和長度。

package com.uzzh;

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

public class UDPThread implements Runnable{
    
    DatagramSocket socket = null;
    DatagramPacket packet = null;
    

    public UDPThread(DatagramSocket socket,DatagramPacket packet) {
        this.socket = socket;
        this.packet = packet;
    }

    @Override
    public void run() {
        String info = null;
        InetAddress address = null;
        int port = 8800;
        byte[] data2 = null;
        DatagramPacket packet2 = null;
        try {
            info = new String(packet.getData(), 0, packet.getLength());
            System.out.println("我是伺服器,客戶端說:"+info);
            
            address = packet.getAddress();
            port = packet.getPort();
            data2 = "我在響應你!".getBytes();
            packet2 = new DatagramPacket(data2, data2.length, address, port);
            socket.send(packet2);
        } catch (IOException e) {
            e.printStackTrace();
        }
        //socket.close();不能關閉         
    }

}

 

2.6 創建伺服器端類

  

package com.uzzh;

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

public class UDPServer {
    public static void main(String[] args) throws IOException {
        DatagramSocket socket = new DatagramSocket(8800);
        DatagramPacket packet = null;
        byte[] data = null;
        int count = 0;
        System.out.println("***伺服器端啟動,等待發送數據***");
        while(true){
            data = new byte[1024];//創建位元組數組,指定接收的數據包的大小
            packet = new DatagramPacket(data, data.length);
            socket.receive(packet);//此方法在接收到數據報之前會一直阻塞
            Thread thread = new Thread(new UDPThread(socket, packet));
            thread.start();
            count++;
            System.out.println("伺服器端被連接過的次數:"+count);
            InetAddress address = packet.getAddress();
            System.out.println("當前客戶端的IP為:"+address.getHostAddress());
            
        }
        
    }
}

 

之前我將new DatagramSocket放入了while迴圈中,報了java.net.BindException: Address already in use: Cannot bind,才知道不能在while中連續創建新的DatagramSocket對象

2.7 創建客戶端類

package com.uzzh;

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

public class UDPClient {
    public static void main(String[] args) throws IOException {
        //定義伺服器的地址,埠號,數據
        InetAddress address = InetAddress.getByName("localhost");
        int port = 8800;
        byte[] data = "用戶名:admin;密碼:123".getBytes();//將字元串轉換為位元組數組
        //創建數據報
        DatagramPacket packet = new DatagramPacket(data, data.length, address, port);
        //創建DatagramSocket,實現數據發送和接收
        DatagramSocket socket = new DatagramSocket();
        //向伺服器端發送數據報
        socket.send(packet);
        
        //接收伺服器響應數據
        byte[] data2 = new byte[1024];
        DatagramPacket packet2 = new DatagramPacket(data2, data2.length);
        socket.receive(packet2);
        String info = new String(data2, 0, packet2.getLength());
        System.out.println("我是客戶端,伺服器說:"+info);
        socket.close();
    }
}

 

2.8 先運行伺服器端,在運行客戶端

2.9 修改客戶端信息,再次運行客戶端

 

 伺服器控制台:伺服器端一直在迴圈等待接收客戶端的數據。

 

三. 總結

  這兩個例子只是簡單的實現了基於TCP和UDP的socket編程,其中像多線程的優先順序等都暫且沒做考慮,不過依然要強調一下,伺服器與多個客戶端進行通信,因為是死迴圈,不設置多線程優先順序,可能會導致運行時速度非常慢,優先順序的範圍1-10,預設為5,我們可以適當降低線程的優先順序,比如thread.setPriority(4);

  對於同一個socket,如果關閉了輸出流比如(pw.close()),則與該輸出流關聯的socket也會關閉,所以一般不需要關閉輸出流,當關閉socket的時候,輸出流也會關閉,直接關閉socket就行。

  在使用TCP通信傳輸信息時,更多是使用對象的形式來傳輸,可以使用ObjectOutputStream對象序列化流來傳遞對象,比如ObjectOutputStream os = new ObjectOutputStream(socket.getOutputStream());User user = new User("admin","123"); os.writeObject(user);

  希望這篇文章能讓你有所獲,麻煩點贊或關註我,謝謝觀看!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 無限級分類是一種設計技巧,在開發中經常使用,例如:網站目錄、部門結構、文章分類。筆者覺得它在對於設計表的層級結構上面發揮很大的作用,比如大家在一些平臺上面, 填寫邀請人,它就是一種上下級的關係,上級會有多個下級,下級又會有自己的分支,大多數都是利用遞歸的思想去實現。話不多說,首先來溫故一下遞歸的實現 ...
  • 如果你還沒有搭建gtest框架,可以參考我之前的博客:http://www.cnblogs.com/jycboy/p/6001153.html。。 1.The first sample: sample1 你把github上的項目導來之後,github地址:https://github.com/goo ...
  • RPC即遠程過程調用,它的實現方式有很多,比如webservice等。框架調多了,煩了,沒激情了,我們就該問自己,這些框架的作用到底是什麼,來找回當初的激情。 一般來說,我們寫的系統就是一個單機系統,一個web伺服器一個資料庫服務,但是當這單台伺服器的處理能力受硬體成本的限制,是不能無限的提升處理性 ...
  • 一、設置一個新的測試項目 在用google test寫測試項目之前,需要先編譯gtest到library庫並將測試與其鏈接。我們為一些流行的構建系統提供了構建文件: msvc/ for Visual Studio, xcode/ for Mac Xcode, make/ for GNU make,  ...
  • 背景說明 最近在工作項目中有下麵一個場景: 使用Node.js的express框架實現了一個文件系統伺服器端,其中有個API用於客戶端上傳文件。客戶端使用Node.js的HttpClient來調用伺服器端的API上傳文件。 客戶端在上傳小文件時沒有任何問題,在上傳大文件時httpClient請求報錯 ...
  • wordpress是用php語言開發的博客平臺,它擴展性強,容易擴展,很適合拿來做二次開發。 1,問題由來 本周五,我在瀏覽公司的網站(基於wordpress開發)時發現,網站首頁上有兩篇文章的縮略圖重覆了,於是我進入網站後臺檢查,想看下是不是某位員工在撰寫文章時不小心這兩篇文章選擇了相同的圖片作為 ...
  • CREATESTRUCT結構CREATESTRUCT結構具有如下形式:typedef struct tagCREATESTRUCT{ LPVOID lpCreateParams; HANDLE hInstance; HMENU hMenu; HWND hwndParent; int cy; int ...
  • 第一種方式: javabean: 1 public class BusLoanInfoShop { 2 private Integer id; 3 private Integer bid; 4 private String shopName; 5 private String platformNam ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...