主頁 > 後端開發 > JAVA Socket編程

JAVA Socket編程

2023-06-03 07:50:18 後端開發

aliases: []
tags   : " "
summary: [基于TCP/IP和UDP協議的Java Socket網路通信編程]
author : [yaenli]
notekey: [20230512-143738]

Socket 網路模型

Socket編程是在TCP/IP、UDP協議上的網路編程,在此之前,先了解下常見的網路模型:

  1. OSI七層模型與TCP模型
    image.png

  2. OSI七層模型詳解(OSI七層模型詳解)
    image.png

Socket就在應用程式的傳輸層和應用層之間的一個抽象層:

image.png

Socket 知識

Socket概述

  • 在計算機網路編程技術中,兩個行程或者說兩臺計算機可以通過一個網路通信連接實作資料的交換,這種通信鏈路的端點就被稱為“套接字”(Socket),
  • Socket 是網路驅動層提供給應用程式的一個介面或者說一種機制,
  • Socket 方便了應用程式訪問通訊協議TCP/IP、UDP ,
  • 我們可以把套接字看成是 電話機,有了套接字,才有了通訊的工具,我們可以把IP地址看成是話號碼, 埠號看成是分機號,

Java中Socket的實作

Socket的底層機制非常復雜,Java平臺提供了一些簡單但是強大的類,可以簡單有效地使用Socket開發通信程式而無須了解底層機制,

java.net包提供了若干支持基于套接字的客戶端/服務器通信的類:

ServerSocket 類用來創建 TCP/IP 服務器端;
Socket 類用來創建 TCP/IP 客戶端;
DatagramSocket 類用來實作 UDP 協議的客戶端和服務器套接字;
DatagramPacket 類用來封裝、處理 UDP 協議的資料包;
InetAddress 類用于封裝IP和DNS等地址資訊,在創建資料報報文和 Socket 物件時,可以使用,

Socket 編程

基于TCP/IP協議的Socket編程

(1)分別使用java.net.Socket和ServerSocket來創建客戶端和服務器端套接字,它們是基于TCP協議進行作業的,作業程序如同打電話的程序,只有雙方都接通了,才能開始通話,
(2)基于TCP創建的套接字叫做 流套接字,Socket通過資料流來完成資料的傳遞作業,
(3)Socket編程中,遵循client-server模型,服務器端相當于一個監聽器,用來監聽埠,

相關類

  • Socket 類:用構造方法創建套接字,并將此套接字連接至指定的主機和埠,
// 常用構造
public Socket(@Nullable  String host, int port ) throws UnknownHostException, IOException ;
public Socket(InetAddress address, int port ) throws IOException ;

// 常用方法
public void connect(SocketAddress host, int timeout) throws IOException;// 將此套接字連接到服務器,并指定一個超時值,
public InetAddress getInetAddress(); // 回傳遠程IP資訊
public int getPort(); // 回傳遠程埠
public int getLocalPort(); // 回傳本地埠
public InputStream getInputStream(); // 獲取輸入流
public OutputStream getOutputStream(); // 獲取輸出流
public void close() throws IOException // 關閉此套接字
  • ServerSocket 類:等待客戶端建立連接,連接建立以后進行通信,
// 常用構造方法
public ServerSocket(int port) throws IOException; // 創建指定埠的服務器套接字
public ServerSocket(int port, int backlog) throws IOException; // 指定最大連接佇列
public ServerSocket(int port, int backlog, InetAddress bindAddr) throws IOException; // 指定系結的本地ip地址

// 常用方法:Socket中的方法都能適用,除此之外,還有以下方法
public Socket accept() throws IOException // (阻塞方法)偵聽連接請求,并回傳一個新的通信套接字,該 Socket 連接到客戶端的 Socket
public void bind(SocketAddress host, int backlog) // 將ServerSocket系結到特定地址

開發流程

image.png

詳細互動程序:

image.png

服務端編程步驟:

  1. 實體化ServerSocket 物件,系結指定埠;
  2. 呼叫 accept(),監聽連接請求(阻塞等待),并回傳通信Socket
  3. Socket 獲取輸出流輸入流,從輸入流中讀取請求資訊,向輸出流中寫入回應資訊;
  4. 關閉資料流和通信套接字,

客戶端編程步驟:

  1. 實體化 Socket 物件,連接到指定服務器端;
  2. Socket 獲取輸出流輸入流,向輸出流中寫入請求資訊,從輸入流中讀取回應資訊;
  3. 關閉資料流和通信套接字,

客戶端和服務器端的互動,采用一問一答的模式,先啟動服務器進入監聽狀態,等待客戶端的連接請求,連接成功以后,客戶端先 “發言”,服務器給予 “回應”,

示例代碼

采用多執行緒的方式,實作一個服務端回應多個客戶端請求,

服務端代碼: 使服務器端Socket一直處于監聽狀態,服務器端每監聽到一個請求,創建一個執行緒物件并啟動,

import java.net.*;
import java.io.*;
 
public class SocketServer {
	
	public static void main(String[] args) {
		ServerSocket serverSocket = null;
		try {
			// 建立一個服務器Socket(ServerSocket)指定埠并開始監聽
			serverSocket = new ServerSocket(8800);
 
			// 監聽一直進行中
			while (true) {
				// 使用accept()方法等待客戶發起通信
				Socket socket = serverSocket.accept();
				new SocketThread(socket).start();
			}
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
}
import java.net.*;
import java.io.*;
 
public class SocketThread extends Thread {

	/*
	 * 	(1)創建服務器端執行緒類,run()方法中實作對一個請求的回應處理,
	 * 	(2)讓服務器端Socket一直處于監聽狀態,
	 * 	(3)服務器端每監聽到一個請求,創建一個執行緒物件并啟動
	 */

	Socket socket = null;
	//每啟動一個執行緒,連接對應的Socket
	public LoginThread(Socket socket) {
		this.socket = socket;
	}
	
	//啟動執行緒,即回應客戶請求
	public void run() {
		InputStream is = null;
		ObjectInputStream ois = null;
		OutputStream os = null;
		try {
			//打開輸入流
			is = socket.getInputStream();
			//反序列化
			ois = new ObjectInputStream(is);
			//獲取客戶端資訊,即從輸入流讀取資訊,DataObject為自定義資料類
			DataObject data = https://www.cnblogs.com/yaenli/p/(DataObject)ois.readObject();
			if(data!=null){
				System.out.println("我是服務端,客戶端傳送資訊為:" + data.getMessage());
			}
			
			//給客戶端一個回應,即向輸出流中寫入資訊
			os = socket.getOutputStream();
			String reply = "服務端接收成功!";
			os.write(reply.getBytes());
		} catch (IOException e) {
			e.printStackTrace();
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}finally{
			try {
				os.close();
				ois.close();
				is.close();
				socket.close();
			} catch (IOException e) {
				e.printStackTrace();
			}			
		}	
	}	
}

客戶端代碼:

import java.net.*;
import java.io.*;
 
public class SocketClient {
	/*
	 * 客戶端通過輸出流向服務器端發送請求資訊
	 * 服務器偵聽客戶端的請求得到一個Socket物件,將這個Socket物件傳遞給執行緒類
	 * 執行緒類通過輸入流獲取客戶端的請求并通過輸出流向客戶端發送回應資訊
	 * 客戶端通過輸入流讀取服務器發送的回應資訊
	 * 
	 */
 
	public static void main(String[] args) {
		
		Socket socket = null;
		OutputStream os = null;
		ObjectOutputStream oos = null;
		InputStream is = null;
		BufferedReader br = null;
		try {
			// 建立客戶端Socket連接,指定服務器的位置為本機以及埠為8800
			socket = new Socket("localhost", 8800);
			// 打開輸出流
			os = socket.getOutputStream();
			// 物件序列化
			oos = new ObjectOutputStream(os);
			// 發送客戶端資訊,即向輸出流中寫入資訊,DataObject為自定義資料類
			DataObject data = https://www.cnblogs.com/yaenli/p/new DataObject("服務端你好,我是客戶端");
			oos.writeObject(data);
			socket.shutdownOutput();
 
			// 接收服務器端的回應,即從輸入流中讀取資訊
			is = socket.getInputStream();
			br = new BufferedReader(new InputStreamReader(is));
			String reply;
			while ((reply = br.readLine()) != null) {
				System.out.println("我是客戶端,服務器的回應為:" + reply);
			}
		} catch (UnknownHostException e) {
			e.printStackTrace();
		} catch (IOException e) {
			e.printStackTrace();
		} finally {
			try {
				br.close();
				is.close();
				oos.close();
				os.close();
				socket.close();
			} catch (IOException e) {
				e.printStackTrace();
			}
		}
	}
}

基于UDP的socket編程

(1)基于TCP的網路通信是安全的,是雙向的,如同打電話,需要先有服務端,建立雙向連接后,才開始資料通信,
(2)基于UDP的套接字就是 資料報套接字,資料報是表示通信的一種報文型別,使用資料報進行通信時無須事先建立連接,只需要指明對方地址,然后將資料送出去,這樣的網路通信是不安全的,所以只應用在如聊天系統、咨詢系統等場合下,
(3)兩端都要先構造好相應的資料包,資料報套接字發送成功之后,就相當于建立了一個虛連接,雙方可以發送資料,
(4)Java中有兩個可使用資料報實作通信的類,即DatagramPacket和DatagramSocket,
(5)DatagramPacket類起到資料容器的作用,DatagramSocket類用于發送或接收DatagramPacket,以此實作資料報通信,

UDP與TCP通信的區別:

TCP UDP
是否連接 面向連接 面向非連接
傳輸可靠性 可靠 不可靠
速度

相關類

  • java.net. DatagramPacket :資料電報包,用于封裝發送的資料,
  • java.net. DatagramSocket :資料電報套接字,不維護連接狀態,不產生輸入/輸出資料流,用于接收和發送DatagramPacket物件封裝好的資料報,

開發流程

image.png

UDP通信的兩個端點程式是平等的,沒有主次之分,甚至它們的代碼都可以完全是一樣的,

接收端編程步驟:

  1. 實體化DatagramSocket 創建資料報套接字,系結到指定埠;
  2. 實體化DatagramPacket 建立要接收的UDP包;
  3. 呼叫DatagramSocket.receive() ,接收UDP包;
  4. 處理接收到的DatagramPacket 資料包,關閉資料報套接字,

發送端編程步驟:

  1. 實體化DatagramSocket 創建資料報套接字,系結到指定埠;
  2. 實體化DatagramPacket 建立要發送的UDP包;
  3. 呼叫DatagramSocket.send() ,發送UDP包;
  4. 關閉資料報套接字,

示例代碼

發送方發送咨詢問題,接收方回應咨詢,

接收端代碼:

import java.net.*;
import java.io.*;
 
public class UDPReceive {
 
	public static void main(String[] args) {
		/*
		 * 接收方實作步驟如下: 
		 * (1)創建DatagramPacket物件,準備接收封裝的資料,
		 * (2)創建DatagramSocket物件,接收資料保存于DatagramPacket物件中,
		 * (3)利用DatagramPacket物件處理資料,
		 */
 
		DatagramSocket ds = null;
		DatagramPacket dp = null;
		DatagramPacket dp_reply = null;
		// 創建DatagramPacket物件,用來準備接收資料
		byte[] buf = new byte[1024];
		dp = new DatagramPacket(buf, 1024);
		try {
			// 創建DatagramSocket物件,接收資料
			ds = new DatagramSocket(8800);
			ds.receive(dp);
			// 顯示接收到的資訊
			String mess = new String(dp.getData(), 0, dp.getLength());
			System.out.println(dp.getAddress().getHostAddress() + "說:" + mess);
			
            // 給發送端回傳資料,需要發送端去接受
			String reply = "接收端:你好,我在,請咨詢!";
			// 創建DatagramPacket物件,封裝資料
			dp_reply = new DatagramPacket(reply.getBytes(),
					reply.getBytes().length, dp.getSocketAddress());
			ds.send(dp_reply);
		} catch (SocketException e) {
			e.printStackTrace();
		} catch (IOException e) {
			e.printStackTrace();
		} finally {
			ds.close();
		}
	}
}

發送端代碼:

import java.net.*;
import java.io.*;
 
public class UDPSend {
	/*
	 * 發送方實作步驟如下: 
	 * (1)獲取本地主機的InetAddress物件, 
	 * (2)創建DatagramPacket物件,封裝要發送的資訊,
	 * (3)利用DatagramSocket物件將DatagramPacket物件資料發送出去,
	 */
 
	public static void main(String[] args) {
		DatagramSocket ds = null;
		InetAddress ia = null;
		String mess = "發送端:你好,我想咨詢一個問題,";
		try {
			// 獲取本地主機地址
			ia = InetAddress.getByName("localhost");
			// 創建DatagramPacket物件,封裝資料
			DatagramPacket dp = new DatagramPacket(mess.getBytes(),
					mess.getBytes().length, ia, 8800);
			// 創建DatagramSocket物件,向服務器發送資料
			ds = new DatagramSocket();
			ds.send(dp);
            
            //接受回傳來的資料,
			byte[] buf = new byte[1024];
			DatagramPacket dpre = new DatagramPacket(buf, buf.length);
			ds.receive(dpre);
			// 顯示接收到的資訊
			String reply = new String(dpre.getData(), 0, dpre.getLength());
			System.out.println(dpre.getAddress().getHostAddress() + "說:"
					+ reply);
		} catch (UnknownHostException e) {
			e.printStackTrace();
		} catch (SocketException e) {
			e.printStackTrace();
		} catch (IOException e) {
			e.printStackTrace();
		} finally {
			ds.close();
		}
	}
}

參考文章

Java網路編程——Socket 編程
Java---Socket編程UDP/TCP-CSDN博客
JAVA進階——Socket編程-CSDN博客
Java Socket實作簡單的Http服務器

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/554127.html

標籤:Java

上一篇:elment UI + EasyExcel 實作 匯入

下一篇:返回列表

標籤雲
其他(160206) Python(38196) JavaScript(25473) Java(18182) C(15236) 區塊鏈(8269) C#(7972) AI(7469) 爪哇(7425) MySQL(7223) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5873) 数组(5741) R(5409) Linux(5346) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4581) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2434) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1980) 功能(1967) Web開發(1951) HtmlCss(1951) C++(1928) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1879) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • JAVA Socket編程

    aliases: [] tags : " " summary: [基于TCP/IP和UDP協議的Java Socket網路通信編程] author : [yaenli] notekey: [20230512-143738] # Socket 網路模型 Socket編程是在TCP/IP、UDP協議上的 ......

    uj5u.com 2023-06-03 07:50:18 more
  • elment UI + EasyExcel 實作 匯入

    前端組件 <hd-flex> <el-dialog v-model="isUploadDialog" width="50%" lock-scroll=false> <el-upload class="upload-demo" drag :action="url" :on-success="succe ......

    uj5u.com 2023-06-03 07:45:08 more
  • 什么是 Apache ActiveMQ?

    # 什么是 ActiveMQ? ActiveMQ 是一種流行的訊息傳遞服務,可促進企業系統中大規模的不同資料。在本 ActiveMQ 教程中,我們概述了 ActiveMQ、它的優點、它的作業原理以及何時應該使用它。 什么是 ActiveMQ? ActiveMQ 是一種流行的開源訊息傳遞服務,它構建在 ......

    uj5u.com 2023-06-03 07:45:03 more
  • 39基于java的酒店管理系統設計與實作

    基于java的酒店管理系統設計與實作,酒店訂票系統,酒店預訂系統,酒店資訊管理系統,app訂房系統設計與實作; ......

    uj5u.com 2023-06-03 07:44:58 more
  • 談談一致性哈希演算法

    一致性哈希演算法是1997年由麻省理工的幾位學者提出的用于解決分布式快取中的熱點問題。大家有沒有發現,我們之前介紹的例如快排之類的演算法是更早的六七十年代,此時分布式還沒有發展起來,大家往往還在提高單機性能。但是九十年代開始,逐漸需要用分布式集群來解決大型問題,相應的演算法研究也就應運而生。在說到一致性哈 ......

    uj5u.com 2023-06-03 07:44:44 more
  • 38基于java的在線商城設計與實作

    基于java的在線商城設計與實作,在線購物平臺,校園購物商城,商品銷售平臺,基于Java的電商平臺;電商平臺,買家和賣家可以在此平臺上進行銷售和交易,節約了大量的線下時間成本,購物車的功能,校園交易平臺等等; ......

    uj5u.com 2023-06-03 07:44:35 more
  • FastJson轉Java對像欄位不區分大小寫

    昨天遇到引數key大小寫不一致導致校驗簽名失敗的問題,查了很長時間才找到原因。看了一下FastJson原始碼,發現JSON.toObject中轉換成物件的時候會忽略大小寫。 所以,當使用了JSON.toObject將json轉成Java物件后,再用JSON.toObject轉成json,key值就變了 ......

    uj5u.com 2023-06-03 07:44:31 more
  • 簡述泛型的基本使用和作用

    # 前言 在上一篇文章中,給大家講解了泛型的概念、作用、使用場景,以及泛型集合、泛型介面和泛型類的用法,但受限于篇幅,并沒有把泛型的內容講解完畢。所以今天我們會繼續學習泛型方法、泛型擦除,以及通配符等的內容,希望大家繼續做好學習的準備哦。 *** 全文大約【**4600】** 字,不說廢話,只講可以 ......

    uj5u.com 2023-06-03 07:44:27 more
  • 為什么說 Go 語言字串是不可變的?

    **原文鏈接:** [為什么說 Go 語言字串是不可變的?](https://mp.weixin.qq.com/s/AOb6AjKwyTwLeAUou0AU-Q) 最近有讀者留言說,平時在寫代碼的程序中,是會對字串進行修改的,但網上都說 Go 語言字串是不可變的,這是為什么呢? 這個問題本身并 ......

    uj5u.com 2023-06-03 07:38:25 more
  • JVM致命錯誤日志詳解

    [toc] 這篇文章是我之前總結的一篇文章,因為整理博客的原因,原有博客已經注銷,但這篇文章對一些讀者很有用,所以現在新瓶裝舊酒重新整理回來分享給大家。 最近一段時間生產環境頻繁出問題,每次都會生成一個hs_err_pid*.log檔案,因為作業內容的原因,在此之前并沒有了解過相關內容,趁此機會學習 ......

    uj5u.com 2023-06-02 10:10:23 more