成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

JAVA 中原生的 socket 通信機制

Eric / 3047人閱讀

摘要:采用標準的傳輸格式,就能進行請求響應了某些特定的框架,可能會有自定義的通信格式。對于這種情況,采用多線程的模型再合適不過。只啟動固定的線程數(shù)來進行處理,既利用了多線程的處理,又控制了系統(tǒng)的資源消耗。在的包中,提供了相應的實現(xiàn)。

JAVA 中原生的 socket 通信機制

摘要:本文屬于原創(chuàng),歡迎轉載,轉載請保留出處:https://github.com/jasonGeng88/blog

當前環(huán)境

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é)操作。

下面以 BufferedReaderPrintWriter 作為輸入輸出的包裝類進行處理。

// 獲取 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

相關文章

  • JAVA NIO 一步步構建I/O多路復用的請求模型

    摘要:為解決這問題,我們發(fā)現(xiàn)元兇處在一線程一請求上,如果一個線程能同時處理多個請求,那么在高并發(fā)下性能上會大大改善。這樣一個線程可以同時發(fā)起多個調用,并且不需要同步等待數(shù)據(jù)就緒。表示當前就緒的事件類型。 JAVA NIO 一步步構建I/O多路復用的請求模型 摘要:本文屬于原創(chuàng),歡迎轉載,轉載請保留出處:https://github.com/jasonGeng88/blog 文章一:JAVA ...

    X_AirDu 評論0 收藏0
  • Object對象你真理解了嗎?

    摘要:無論在中出現(xiàn)什么,都可以認為它是對象除了八大基本數(shù)據(jù)類型。讓當前線程等待某個對象的鎖,當然應該通過這個對象來操作了。但是要注意的是方法調用后,被喚醒的線程不會立馬獲得到鎖對象。主要的區(qū)別在于在釋放同時,釋放了對象鎖的控制。 前言 五一回家又斷更了一個放假時間了~~~ 只有光頭才能變強 回顧前面: ThreadLocal就是這么簡單 多線程三分鐘就可以入個門了! 多線程基礎必要知識點!...

    anquan 評論0 收藏0
  • Android IPC機制(五)用Socket實現(xiàn)跨進程聊天程序

    摘要:簡介簡介也稱作套接字,是在應用層和傳輸層之間的一個抽象層,它把層復雜的操作抽象為幾個簡單的接口供應用層調用以實現(xiàn)進程在網(wǎng)絡中通信。它分為流式套接字和數(shù)據(jù)包套接字,分別對應網(wǎng)絡傳輸控制層的和協(xié)議。1.Socket簡介 Socket也稱作套接字,是在應用層和傳輸層之間的一個抽象層,它把TCP/IP層復雜的操作抽象為幾個簡單的接口供應用層調用以實現(xiàn)進程在網(wǎng)絡中通信。它分為流式套接字和數(shù)據(jù)包套接字,...

    lufficc 評論0 收藏0
  • Java學習筆記7-網(wǎng)絡編程

    摘要:網(wǎng)絡編程是指編寫運行在多個設備計算機的程序,這些設備都通過網(wǎng)絡連接起來。通常用于互聯(lián)網(wǎng)協(xié)議,被稱。編程套接字使用提供了兩臺計算機之間的通信機制。客戶端程序創(chuàng)建一個套接字,并嘗試連接服務器的套接字。 網(wǎng)絡編程是指編寫運行在多個設備(計算機)的程序,這些設備都通過網(wǎng)絡連接起來。 網(wǎng)絡編程是指編寫運行在多個設備(計算機)的程序,這些設備都通過網(wǎng)絡連接起來。 java.net 包中 J2SE ...

    lufficc 評論0 收藏0
  • Java 遠程通訊技術及原理分析

    摘要:對于與而言,則可以看做是消息傳遞技術的一種衍生或封裝。在生產者通知消費者時,傳遞的往往是消息或事件,而非生產者自身。通過消息路由,我們可以配置路由規(guī)則指定消息傳遞的路徑,以及指定具體的消費者消費對應的生產者。采用和來進行遠程對象的通訊。 消息模式 歸根結底,企業(yè)應用系統(tǒng)就是對數(shù)據(jù)的處理,而對于一個擁有多個子系統(tǒng)的企業(yè)應用系統(tǒng)而言,它的基礎支撐無疑就是對消息的處理。與對象不同,消息本質上...

    rozbo 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<