Netty 之傳統(tǒng) BIO菠劝、NIO 編程

說明

系列文章:http://www.reibang.com/p/594441fb9c9e

本文完全參考自《Netty權(quán)威指南(第2版)》,李林峰著叹螟。

傳統(tǒng) BIO 編程是什么樣的魄宏?

基本的網(wǎng)絡(luò)編程模型是Client/Server秸侣,即兩個進程間相互通信,其中服務(wù)端提供位置信息(IP地址和端口號)宠互,客戶端通過連接向服務(wù)器監(jiān)聽的地址發(fā)起連接請求味榛,通過三次握手建立連接,之后雙方就可以通過網(wǎng)絡(luò)套接字(socket)進行通信予跌。

Java的傳統(tǒng)同步阻塞模型中搏色,ServerSocket負責(zé)綁定IP地址,啟動監(jiān)聽端口券册;Socket負責(zé)發(fā)起連接频轿。之后雙方通過輸入和輸出流進行同步阻塞式通信垂涯。

BIO 通信模型

對于每個客戶端,服務(wù)端都要新建一個線程航邢。

當(dāng)客戶端并發(fā)訪問量增加后耕赘,服務(wù)端的線程個數(shù)和客戶端并發(fā)訪問數(shù)量呈1:1的關(guān)系,當(dāng)線程數(shù)膨脹后膳殷,系統(tǒng)的性能將急劇下降操骡。

示例

服務(wù)端在接收到字符串QUERY TIME ORDER后,返回當(dāng)前日期給客戶端赚窃。

TimeServer 源碼

public class TimeServer {

    public static void main(String[] args) throws IOException {
        int port = 8080;
        ServerSocket serverSocket = null;
        try {
            serverSocket = new ServerSocket(port);
            System.out.println("The time server is start in port : " + port);
            Socket socket = null;
            while (true) {
                socket = serverSocket.accept();
                new Thread(new TimeServerHandler(socket)).start();
            }
        } finally {
            if (serverSocket != null) {
                System.out.println("The time server close");
                serverSocket.close();
                serverSocket = null;
            }
        }
    }
}
public class TimeServerHandler implements Runnable {

    private Socket socket;

    public TimeServerHandler(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        BufferedReader in = null;
        PrintWriter out = null;

        try {
            in = new BufferedReader(new InputStreamReader(this.socket.getInputStream()));
            out = new PrintWriter(this.socket.getOutputStream(), true);
            String currentTime = null;
            String body = null;
            while (true) {
                body = in.readLine();
                if (body == null) {
                    break;
                }
                System.out.println("The time server receive order : " + body);
                currentTime = "QUERY TIME ORDER".equalsIgnoreCase(body) ? new Date().toString() : "BAD ORDER";
                out.println(currentTime);
            }
        } catch (Exception e) {
            if (in != null) {
                try {
                    in.close();
                } catch (IOException e1) {
                    e1.printStackTrace();
                }
            }
            if (out != null) {
                out.close();
                out = null;
            }
            if (this.socket != null) {
                try {
                    this.socket.close();
                } catch (IOException e1) {
                    e1.printStackTrace();
                }
                this.socket = null;
            }
        }
    }
}

TimeClient 源碼

public class TimeClient {
    public static void main(String[] args) throws Exception {
        int port = 8080;
        Socket socket = null;
        BufferedReader in = null;
        PrintWriter out = null;
        try {
            socket = new Socket("127.0.0.1", port);
            in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            out = new PrintWriter(socket.getOutputStream(), true);
            out.println("QUERY TIME ORDER");
            System.out.println("SEND TO SERVER SUCCESSED");
            String resp = in.readLine();
            System.out.println("Now is : " + resp);
        } catch (Exception e) {
            // TODO: handle exception
        } finally {
            if (out != null) {
                out.close();
                out = null;
            }
            if (in != null) {
                in.close();
                in = null;
            }
            if (socket != null) {
                socket.close();
                socket = null;
            }
        }
    }
}

輸出結(jié)果

The time server is start in port : 8080
The time server receive order : QUERY TIME ORDER
The time server receive order : QUERY TIME ORDER

線程堆棧查看

通過jstack命令册招,查看當(dāng)前堆棧信息:

"main" #1 prio=5 os_prio=31 tid=0x00007ffed1805000 nid=0x1c03 runnable [0x0000700007de7000]
   java.lang.Thread.State: RUNNABLE
    at java.net.PlainSocketImpl.socketAccept(Native Method)
    at java.net.AbstractPlainSocketImpl.accept(AbstractPlainSocketImpl.java:409)
    at java.net.ServerSocket.implAccept(ServerSocket.java:545)
    at java.net.ServerSocket.accept(ServerSocket.java:513)
    at bio.TimeServer.main(TimeServer.java:18)

我們可以看到,代碼確實阻塞在accept操作上勒极。

偽異步 I/O 編程

同步阻塞I/O當(dāng)海量并發(fā)接入時是掰,會導(dǎo)致線程耗盡『又剩可以對線程模型進行優(yōu)化:后端通過一個線程池來處理多個客戶端的請求接入。

但是偽異步I/O也存在很多問題震叙,因為它的本質(zhì)仍然是同步阻塞掀鹅。這意味著當(dāng)對方發(fā)送請求或者應(yīng)答消息比較緩慢,或者網(wǎng)絡(luò)傳輸較慢時媒楼,讀取輸入流一方的通信線程會被長時間阻塞乐尊;在此期間,其他接入消息只能在消息隊列中排隊划址。

NIO 編程

首先問自己一個問題扔嵌,什么才是NIO編程?對于NIO夺颤,官方的說法是:New I/O痢缎;但是更多的人喜歡稱之為Non-block I/O(非阻塞I/O)。

BIO中的Socket類和ServerSocket類世澜,對應(yīng)于NIO中的SocketChannel和ServerSocketChannel兩種不同的套接字通道實現(xiàn)独旷,這兩種套接字都支持阻塞和非阻塞兩種模式。

  • 阻塞模式:使用非常簡單寥裂,但是性能和可靠性都不好嵌洼;
  • 非阻塞模式:使用復(fù)雜,性能和可靠性好封恰。

NIO 類庫簡介

緩沖區(qū) Buffer

在面向流的I/O中麻养,可以直接讀取或?qū)懭霐?shù)據(jù)至Stream對象中;在NIO中诺舔,所有數(shù)據(jù)都是通過緩沖區(qū)處理的:讀取數(shù)據(jù)時鳖昌,直接讀到緩沖區(qū)备畦;寫入數(shù)據(jù)時,寫入到緩沖區(qū)遗遵。任何時候訪問NIO中的數(shù)據(jù)萍恕,都是通過緩沖區(qū)進行的。

通道 Channel

傳統(tǒng)的只有一個方向(InputStream或者OutputStream)车要,而通道可以用于讀允粤、寫或二者同時進行。

多路復(fù)用器 Selector

提供選擇已經(jīng)就緒的任務(wù)的能力翼岁。Selector會不斷輪詢注冊在其上的Channel类垫,如果某個Channel上面發(fā)生讀寫事件,這個Channel處于就緒狀態(tài)琅坡,會被Selector輪詢處理悉患,然后通過SelectionKey可以獲取就緒Channel的集合,進行后續(xù)的I/O操作榆俺。


對于NIO的編程十分繁瑣售躁,就不作介紹,在接下來的文章中直接使用Netty開發(fā)茴晋。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末陪捷,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子诺擅,更是在濱河造成了極大的恐慌市袖,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件烁涌,死亡現(xiàn)場離奇詭異苍碟,居然都是意外死亡,警方通過查閱死者的電腦和手機撮执,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門微峰,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人抒钱,你說我怎么就攤上這事县忌。” “怎么了继效?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵症杏,是天一觀的道長。 經(jīng)常有香客問我瑞信,道長厉颤,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任凡简,我火速辦了婚禮逼友,結(jié)果婚禮上精肃,老公的妹妹穿的比我還像新娘。我一直安慰自己帜乞,他們只是感情好司抱,可當(dāng)我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著黎烈,像睡著了一般习柠。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上照棋,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天资溃,我揣著相機與錄音,去河邊找鬼烈炭。 笑死溶锭,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的符隙。 我是一名探鬼主播趴捅,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼霹疫!你這毒婦竟也來了拱绑?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤更米,失蹤者是張志新(化名)和其女友劉穎欺栗,沒想到半個月后毫痕,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體征峦,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年消请,在試婚紗的時候發(fā)現(xiàn)自己被綠了栏笆。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡臊泰,死狀恐怖蛉加,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情缸逃,我是刑警寧澤针饥,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站需频,受9級特大地震影響丁眼,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜昭殉,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一苞七、第九天 我趴在偏房一處隱蔽的房頂上張望藐守。 院中可真熱鬧,春花似錦蹂风、人聲如沸卢厂。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽慎恒。三九已至,卻和暖如春礁阁,著一層夾襖步出監(jiān)牢的瞬間巧号,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工姥闭, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留丹鸿,地道東北人。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓棚品,卻偏偏與公主長得像靠欢,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子铜跑,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,864評論 2 354

推薦閱讀更多精彩內(nèi)容