摘要:采用標準的傳輸格式,就能進行請求響應了某些特定的框架,可能會有自定義的通信格式。對于這種情況,采用多線程的模型再合適不過。只啟動固定的線程數(shù)來進行處理,既利用了多線程的處理,又控制了系統(tǒng)的資源消耗。在的包中,提供了相應的實現(xiàn)。
JAVA 中原生的 socket 通信機制
當前環(huán)境摘要:本文屬于原創(chuàng),歡迎轉載,轉載請保留出處:https://github.com/jasonGeng88/blog
jdk == 1.8
知識點socket 的連接處理
IO 輸入、輸出流的處理
請求數(shù)據(jù)格式處理
請求模型優(yōu)化
場景今天,和大家聊一下 JAVA 中的 socket 通信問題。這里采用最簡單的一請求一響應模型為例,假設我們現(xiàn)在需要向 baidu 站點進行通信。我們用 JAVA 原生的 socket 該如何實現(xiàn)。
建立 socket 連接首先,我們需要建立 socket 連接(核心代碼)
import java.net.InetSocketAddress; import java.net.Socket; import java.net.SocketAddress; // 初始化 socket Socket socket = new Socket(); // 初始化遠程連接地址 SocketAddress remote = new InetSocketAddress(host, port); // 建立連接 socket.connect(remote);處理 socket 輸入輸出流
成功建立 socket 連接后,我們就能獲得它的輸入輸出流,通信的本質是對輸入輸出流的處理。通過輸入流,讀取網(wǎng)絡連接上傳來的數(shù)據(jù),通過輸出流,將本地的數(shù)據(jù)傳出給遠端。
socket 連接實際與處理文件流有點類似,都是在進行 IO 操作。
獲取輸入、輸出流代碼如下:
// 輸入流 InputStream in = socket.getInputStream(); // 輸出流 OutputStream out = socket.getOutputStream();
關于 IO 流的處理,我們一般會用相應的包裝類來處理 IO 流,如果直接處理的話,我們需要對 byte[] 進行操作,而這是相對比較繁瑣的。如果采用包裝類,我們可以直接以string、int等類型進行處理,簡化了 IO 字節(jié)操作。
下面以 BufferedReader 與 PrintWriter 作為輸入輸出的包裝類進行處理。
// 獲取 socket 輸入流 private BufferedReader getReader(Socket socket) throws IOException { InputStream in = socket.getInputStream(); return new BufferedReader(new InputStreamReader(in)); } // 獲取 socket 輸出流 private PrintWriter getWriter(Socket socket) throws IOException { OutputStream out = socket.getOutputStream(); return new PrintWriter(new OutputStreamWriter(out)); }數(shù)據(jù)請求與響應
有了 socket 連接、IO 輸入輸出流,下面就該向發(fā)送請求數(shù)據(jù),以及獲取請求的響應結果。
因為有了 IO 包裝類的支持,我們可以直接以字符串的格式進行傳輸,由包裝類幫我們將數(shù)據(jù)裝換成相應的字節(jié)流。
因為我們與 baidu 站點進行的是 HTTP 訪問,所有我們不需要額外定義輸出格式。采用標準的 HTTP 傳輸格式,就能進行請求響應了(某些特定的 RPC 框架,可能會有自定義的通信格式)。
請求的數(shù)據(jù)內容處理如下:
public class HttpUtil { public static String compositeRequest(String host){ return "GET / HTTP/1.1 " + "Host: " + host + " " + "User-Agent: curl/7.43.0 " + "Accept: */* "; } }
發(fā)送請求數(shù)據(jù)代碼如下:
// 發(fā)起請求 PrintWriter writer = getWriter(socket); writer.write(HttpUtil.compositeRequest(host)); writer.flush();
接收響應數(shù)據(jù)代碼如下:
// 讀取響應 String msg; BufferedReader reader = getReader(socket); while ((msg = reader.readLine()) != null){ System.out.println(msg); }結果展示
至此,講完了原生 socket 下的創(chuàng)建連接、發(fā)送請求與接收響應的所有核心代碼。
完整代碼如下:
import java.io.*; import java.net.InetSocketAddress; import java.net.Socket; import java.net.SocketAddress; import com.test.network.util.HttpUtil; public class SocketHttpClient { public void start(String host, int port) { // 初始化 socket Socket socket = new Socket(); try { // 設置 socket 連接 SocketAddress remote = new InetSocketAddress(host, port); socket.setSoTimeout(5000); socket.connect(remote); // 發(fā)起請求 PrintWriter writer = getWriter(socket); System.out.println(HttpUtil.compositeRequest(host)); writer.write(HttpUtil.compositeRequest(host)); writer.flush(); // 讀取響應 String msg; BufferedReader reader = getReader(socket); while ((msg = reader.readLine()) != null){ System.out.println(msg); } } catch (IOException e) { e.printStackTrace(); } finally { try { socket.close(); } catch (IOException e) { e.printStackTrace(); } } } private BufferedReader getReader(Socket socket) throws IOException { InputStream in = socket.getInputStream(); return new BufferedReader(new InputStreamReader(in)); } private PrintWriter getWriter(Socket socket) throws IOException { OutputStream out = socket.getOutputStream(); return new PrintWriter(new OutputStreamWriter(out)); } }
下面,我們通過實例化一個客戶端,來展示 socket 通信的結果。
public class Application { public static void main(String[] args) { new SocketHttpClient().start("www.baidu.com", 80); } }
結果輸出:
請求模型優(yōu)化這種方式,雖然實現(xiàn)功能沒什么問題。但是我們細看,發(fā)現(xiàn)在 IO 寫入與讀取過程,是發(fā)生了 IO 阻塞的情況。即:
// 會發(fā)生 IO 阻塞 writer.write(HttpUtil.compositeRequest(host)); reader.readLine();
所以如果要同時請求10個不同的站點,如下:
public class SingleThreadApplication { public static void main(String[] args) { // HttpConstant.HOSTS 為 站點集合 for (String host: HttpConstant.HOSTS) { new SocketHttpClient().start(host, HttpConstant.PORT); } } }
它一定是第一個請求響應結束后,才會發(fā)起下一個站點處理。
這在服務端更明顯,雖然這里的代碼是客戶端連接,但是具體的操作和服務端是差不多的。請求只能一個個串行處理,這在響應時間上肯定不能達標。
多線程處理
有人覺得這根本不是問題,JAVA 是多線程的編程語言。對于這種情況,采用多線程的模型再合適不過。
public class MultiThreadApplication { public static void main(String[] args) { for (final String host: HttpConstant.HOSTS) { Thread t = new Thread(new Runnable() { public void run() { new SocketHttpClient().start(host, HttpConstant.PORT); } }); t.start(); } } }
這種方式起初看起來挺有用的,但并發(fā)量一大,應用會起很多的線程。都知道,在服務器上,每一個線程實際都會占據(jù)一個文件句柄。而服務器上的句柄數(shù)是有限的,而且大量的線程,造成的線程間切換的消耗也會相當?shù)拇?。所以這種方式在并發(fā)量大的場景下,一定是承載不住的。
多線程 + 線程池 處理
既然線程太多不行,那我們控制一下線程創(chuàng)建的數(shù)目不就行了。只啟動固定的線程數(shù)來進行 socket 處理,既利用了多線程的處理,又控制了系統(tǒng)的資源消耗。
public class ThreadPoolApplication { public static void main(String[] args) { ExecutorService executorService = Executors.newFixedThreadPool(8); for (final String host: HttpConstant.HOSTS) { Thread t = new Thread(new Runnable() { public void run() { new SocketHttpClient().start(host, HttpConstant.PORT); } }); executorService.submit(t); new SocketHttpClient().start(host, HttpConstant.PORT); } } }
關于啟動的線程數(shù),一般 CPU 密集型會設置在 N+1(N為CPU核數(shù)),IO 密集型設置在 2N + 1。
這種方式,看起來是最優(yōu)的了。那有沒有更好的呢,如果一個線程能同時處理多個 socket 連接,并且在每個 socket 輸入輸出數(shù)據(jù)沒有準備好的情況下,不進行阻塞,那是不是更優(yōu)呢。這種技術叫做“IO多路復用”。在 JAVA 的 nio 包中,提供了相應的實現(xiàn)。
后續(xù)JAVA 中是如何實現(xiàn) IO多路復用
Netty 下的實現(xiàn)異步請求的
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/70203.html
摘要:為解決這問題,我們發(fā)現(xiàn)元兇處在一線程一請求上,如果一個線程能同時處理多個請求,那么在高并發(fā)下性能上會大大改善。這樣一個線程可以同時發(fā)起多個調用,并且不需要同步等待數(shù)據(jù)就緒。表示當前就緒的事件類型。 JAVA NIO 一步步構建I/O多路復用的請求模型 摘要:本文屬于原創(chuàng),歡迎轉載,轉載請保留出處:https://github.com/jasonGeng88/blog 文章一:JAVA ...
摘要:無論在中出現(xiàn)什么,都可以認為它是對象除了八大基本數(shù)據(jù)類型。讓當前線程等待某個對象的鎖,當然應該通過這個對象來操作了。但是要注意的是方法調用后,被喚醒的線程不會立馬獲得到鎖對象。主要的區(qū)別在于在釋放同時,釋放了對象鎖的控制。 前言 五一回家又斷更了一個放假時間了~~~ 只有光頭才能變強 回顧前面: ThreadLocal就是這么簡單 多線程三分鐘就可以入個門了! 多線程基礎必要知識點!...
摘要:簡介簡介也稱作套接字,是在應用層和傳輸層之間的一個抽象層,它把層復雜的操作抽象為幾個簡單的接口供應用層調用以實現(xiàn)進程在網(wǎng)絡中通信。它分為流式套接字和數(shù)據(jù)包套接字,分別對應網(wǎng)絡傳輸控制層的和協(xié)議。1.Socket簡介 Socket也稱作套接字,是在應用層和傳輸層之間的一個抽象層,它把TCP/IP層復雜的操作抽象為幾個簡單的接口供應用層調用以實現(xiàn)進程在網(wǎng)絡中通信。它分為流式套接字和數(shù)據(jù)包套接字,...
摘要:網(wǎng)絡編程是指編寫運行在多個設備計算機的程序,這些設備都通過網(wǎng)絡連接起來。通常用于互聯(lián)網(wǎng)協(xié)議,被稱。編程套接字使用提供了兩臺計算機之間的通信機制。客戶端程序創(chuàng)建一個套接字,并嘗試連接服務器的套接字。 網(wǎng)絡編程是指編寫運行在多個設備(計算機)的程序,這些設備都通過網(wǎng)絡連接起來。 網(wǎng)絡編程是指編寫運行在多個設備(計算機)的程序,這些設備都通過網(wǎng)絡連接起來。 java.net 包中 J2SE ...
摘要:對于與而言,則可以看做是消息傳遞技術的一種衍生或封裝。在生產者通知消費者時,傳遞的往往是消息或事件,而非生產者自身。通過消息路由,我們可以配置路由規(guī)則指定消息傳遞的路徑,以及指定具體的消費者消費對應的生產者。采用和來進行遠程對象的通訊。 消息模式 歸根結底,企業(yè)應用系統(tǒng)就是對數(shù)據(jù)的處理,而對于一個擁有多個子系統(tǒng)的企業(yè)應用系統(tǒng)而言,它的基礎支撐無疑就是對消息的處理。與對象不同,消息本質上...
閱讀 1794·2023-04-25 22:42
閱讀 2218·2021-09-22 15:16
閱讀 3495·2021-08-30 09:44
閱讀 493·2019-08-29 16:44
閱讀 3316·2019-08-29 16:20
閱讀 2521·2019-08-29 16:12
閱讀 3395·2019-08-29 16:07
閱讀 673·2019-08-29 15:08