九色国产,午夜在线视频,新黄色网址,九九色综合,天天做夜夜做久久做狠狠,天天躁夜夜躁狠狠躁2021a,久久不卡一区二区三区

打開APP
userphoto
未登錄

開通VIP,暢享免費(fèi)電子書等14項(xiàng)超值服

開通VIP
Netty那點(diǎn)事(一)概述

Netty和Mina是Java世界非常知名的通訊框架。它們都出自同一個作者,Mina誕生略早,屬于Apache基金會,而Netty開始在Jboss名下,后來出來自立門戶netty.io。關(guān)于Mina已有@FrankHui的Mina系列文章,我正好最近也要做一些網(wǎng)絡(luò)方面的開發(fā),就研究一下Netty的源碼,順便分享出來了。

Netty目前有兩個分支:4.x和3.x。4.0分支重寫了很多東西,并對項(xiàng)目進(jìn)行了分包,規(guī)模比較龐大,入手會困難一些,而3.x版本則已經(jīng)被廣泛使用。本系列文章針對netty 3.7.0 final。3.x和4.0的區(qū)別可以參考這篇文章:http://www.oschina.net/translate/netty-4-0-new-and-noteworthy?print。

起:Netty是什么

大概用Netty的,無論新手還是老手,都知道它是一個“網(wǎng)絡(luò)通訊框架”。所謂框架,基本上都是一個作用:基于底層API,提供更便捷的編程模型。那么"通訊框架"到底做了什么事情呢?回答這個問題并不太容易,我們不妨反過來看看,不使用netty,直接基于NIO編寫網(wǎng)絡(luò)程序,你需要做什么(以Server端TCP連接為例,這里我們使用Reactor模型):

  1. 監(jiān)聽端口,建立Socket連接
  2. 建立線程,處理內(nèi)容
    1. 讀取Socket內(nèi)容,并對協(xié)議進(jìn)行解析
    2. 進(jìn)行邏輯處理
    3. 回寫響應(yīng)內(nèi)容
    4. 如果是多次交互的應(yīng)用(SMTP、FTP),則需要保持連接多進(jìn)行幾次交互
  3. 關(guān)閉連接

建立線程是一個比較耗時(shí)的操作,同時(shí)維護(hù)線程本身也有一些開銷,所以我們會需要多線程機(jī)制,幸好JDK已經(jīng)有很方便的多線程框架了,這里我們不需要花很多心思。

此外,因?yàn)門CP連接的特性,我們還要使用連接池來進(jìn)行管理:

  1. 建立TCP連接是比較耗時(shí)的操作,對于頻繁的通訊,保持連接效果更好
  2. 對于并發(fā)請求,可能需要建立多個連接
  3. 維護(hù)多個連接后,每次通訊,需要選擇某一可用連接
  4. 連接超時(shí)和關(guān)閉機(jī)制

想想就覺得很復(fù)雜了!實(shí)際上,基于NIO直接實(shí)現(xiàn)這部分東西,即使是老手也容易出現(xiàn)錯誤,而使用Netty之后,你只需要關(guān)注邏輯處理部分就可以了。

承:體驗(yàn)Netty

這里我們引用Netty的example包里的一個例子,一個簡單的EchoServer,它接受客戶端輸入,并將輸入原樣返回。其主要代碼如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public void run() {
    // Configure the server.
    ServerBootstrap bootstrap = new ServerBootstrap(
            new NioServerSocketChannelFactory(
                    Executors.newCachedThreadPool(),
                    Executors.newCachedThreadPool()));
    // Set up the pipeline factory.
    bootstrap.setPipelineFactory(new ChannelPipelineFactory() {
        public ChannelPipeline getPipeline() throws Exception {
            return Channels.pipeline(new EchoServerHandler());
        }
    });
    // Bind and start to accept incoming connections.
    bootstrap.bind(new InetSocketAddress(port));
}

這里EchoServerHandler是其業(yè)務(wù)邏輯的實(shí)現(xiàn)者,大致代碼如下:

1
2
3
4
5
6
7
8
9
public class EchoServerHandler extends SimpleChannelUpstreamHandler {
    @Override
    public void messageReceived(
            ChannelHandlerContext ctx, MessageEvent e) {
        // Send back the received message to the remote peer.
        e.getChannel().write(e.getMessage());
    }
}

還是挺簡單的,不是嗎?

轉(zhuǎn):Netty背后的事件驅(qū)動機(jī)制

完成了以上一段代碼,我們算是與Netty進(jìn)行了第一次親密接觸。如果想深入學(xué)習(xí)呢?

首先推薦Netty的官方User Guide:http://netty.io/3.7/guide/。其次,閱讀源碼是了解一個開源工具非常好的手段,但是Java世界的框架大多追求大而全,功能完備,如果逐個閱讀,難免迷失方向,Netty也并不例外。相反,抓住幾個重點(diǎn)對象,理解其領(lǐng)域概念及設(shè)計(jì)思想,從而理清其脈絡(luò),相當(dāng)于打通了任督二脈,以后的閱讀就不再困難了。

理解Netty的關(guān)鍵點(diǎn)在哪呢?我覺得,除了NIO的相關(guān)知識,另一個就是事件驅(qū)動的設(shè)計(jì)思想。什么叫事件驅(qū)動?我們回頭看看EchoServerHandler的代碼,其中的參數(shù):public void messageReceived(ChannelHandlerContext ctx, MessageEvent e),MessageEvent就是一個事件。這個事件攜帶了一些信息,例如這里e.getMessage()就是消息的內(nèi)容,而EchoServerHandler則描述了處理這種事件的方式。一旦某個事件觸發(fā),相應(yīng)的Handler則會被調(diào)用,并進(jìn)行處理。這種事件機(jī)制在UI編程里廣泛應(yīng)用,而Netty則將其應(yīng)用到了網(wǎng)絡(luò)編程領(lǐng)域。

在Netty里,所有事件都來自ChannelEvent接口,這些事件涵蓋監(jiān)聽端口、建立連接、讀寫數(shù)據(jù)等網(wǎng)絡(luò)通訊的各個階段。而事件的處理者就是ChannelHandler,這樣,不但是業(yè)務(wù)邏輯,連網(wǎng)絡(luò)通訊流程中底層的處理,都可以通過實(shí)現(xiàn)ChannelHandler來完成了。事實(shí)上,Netty內(nèi)部的連接處理、協(xié)議編解碼、超時(shí)等機(jī)制,都是通過handler完成的。當(dāng)博主弄明白其中的奧妙時(shí),不得不佩服這種設(shè)計(jì)!

下圖描述了Netty進(jìn)行事件處理的流程。Channel是連接的通道,是ChannelEvent的產(chǎn)生者,而ChannelPipeline可以理解為ChannelHandler的集合。

合:開啟Netty源碼之門

理解了Netty的事件驅(qū)動機(jī)制,我們現(xiàn)在可以來研究Netty的各個模塊了。Netty的包結(jié)構(gòu)如下:

1
2
3
4
5
6
7
8
9
10
11
org
└── jboss
    └── netty
        ├── bootstrap 配置并啟動服務(wù)的類
        ├── buffer 緩沖相關(guān)類,對NIO Buffer做了一些封裝
        ├── channel 核心部分,處理連接
        ├── container 連接其他容器的代碼
        ├── example 使用示例
        ├── handler 基于handler的擴(kuò)展部分,實(shí)現(xiàn)協(xié)議編解碼等附加功能
        ├── logging 日志
        └── util 工具類

在這里面,channelhandler兩部分比較復(fù)雜。我們不妨與Netty官方的結(jié)構(gòu)圖對照一下,來了解其功能。

具體的解釋可以看這里:http://netty.io/3.7/guide/#architecture。圖中可以看到,除了之前說到的事件驅(qū)動機(jī)制之外,Netty的核心功能還包括兩部分:

  • Zero-Copy-Capable Rich Byte Buffer

    零拷貝的Buffer。為什么叫零拷貝?因?yàn)樵跀?shù)據(jù)傳輸時(shí),最終處理的數(shù)據(jù)會需要對單個傳輸層的報(bào)文,進(jìn)行組合或者拆分。NIO原生的ByteBuffer要做到這件事,需要對ByteBuffer內(nèi)容進(jìn)行拷貝,產(chǎn)生新的ByteBuffer,而Netty通過提供Composite(組合)和Slice(切分)兩種Buffer來實(shí)現(xiàn)零拷貝。這部分代碼在org.jboss.netty.buffer包中。

  • Universal Communication API

    統(tǒng)一的通訊API。因?yàn)镴ava的Old I/O和New I/O,使用了互不兼容的API,而Netty則提供了統(tǒng)一的API(org.jboss.netty.channel.Channel)來封裝這兩種I/O模型。這部分代碼在org.jboss.netty.channel包中。

此外,Protocol Support功能通過handler機(jī)制實(shí)現(xiàn)。

接下來的文章,我們會根據(jù)模塊,詳細(xì)的對Netty源碼進(jìn)行分析。

最后附上Netty那點(diǎn)事系列文章/代碼的Github地址:https://github.com/code4craft/netty-learning

參考資料:

Netty那點(diǎn)事系列文章索引

本站僅提供存儲服務(wù),所有內(nèi)容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點(diǎn)擊舉報(bào)
打開APP,閱讀全文并永久保存 查看更多類似文章
猜你喜歡
類似文章
Netty了解與小試
Netty
Netty 3.1 中文用戶手冊 | Java & Game
netty源代碼解讀
一步一圖,走進(jìn) Netty 的世界
netty的timeout
更多類似文章 >>
生活服務(wù)
熱點(diǎn)新聞
分享 收藏 導(dǎo)長圖 關(guān)注 下載文章
綁定賬號成功
后續(xù)可登錄賬號暢享VIP特權(quán)!
如果VIP功能使用有故障,
可點(diǎn)擊這里聯(lián)系客服!

聯(lián)系客服