Netty自動重連機制的Demo

本博客 貓叔的博客金麸,轉(zhuǎn)載請申明出處

閱讀本文約 “4分鐘”

適讀人群:Java-Netty 初級

Netty自動重連機制

版本:netty 4.1.*

申明:本文旨在重新分享討論Netty官方相關(guān)案例,添加部分個人理解與要點解析矿瘦。

這個是InChat的案例地址篡悟,里面補充了詳細的注釋劫樟,比起官方會容易看一點也糊。

官方案例地址:https://netty.io/4.1/xref/io/netty/example/uptime/package-summary.html

正文

  • UptimeClient(客戶端)
  • UptimeClientHandler
  • UptimeServer(服務端)
  • UptimeServerHandler

要點介紹

一個對Channel尚未執(zhí)行讀、寫或兩次操作的觸發(fā)器

屬性 含義
readerIdleTime 在IdleStateEvent其狀態(tài)IdleState.READER_IDLE 時的指定時間段沒有執(zhí)行讀操作將被觸發(fā)逮壁。指定0禁用。
writerIdleTime 在IdleStateEvent其狀態(tài)IdleState.WRITER_IDLE 時的指定時間段沒有執(zhí)行寫操作將被觸發(fā)粮宛。指定0禁用窥淆。
allIdleTime 一個IdleStateEvent其狀態(tài)IdleState.ALL_IDLE 時的時間在規(guī)定的時間進行讀取和寫入都將被觸發(fā)。指定0禁用窟勃。

如下一個在沒有信息時發(fā)送ping消息祖乳,且30秒沒有入站信息則關(guān)閉連接

public class MyChannelInitializer extends ChannelInitializer<Channel> {
      @Override
     public void initChannel(Channel channel) {
         channel.pipeline().addLast("idleStateHandler", new IdleStateHandler(60, 30, 0));
         channel.pipeline().addLast("myHandler", new MyHandler());
     }
 }

 // Handler should handle the IdleStateEvent triggered by IdleStateHandler.
 public class MyHandler extends ChannelDuplexHandler {
      @Override
     public void userEventTriggered(ChannelHandlerContext ctx, Object evt) throws Exception {
         if (evt instanceof IdleStateEvent) {
             IdleStateEvent e = (IdleStateEvent) evt;
             if (e.state() == IdleState.READER_IDLE) {
                 ctx.close();
             } else if (e.state() == IdleState.WRITER_IDLE) {
                 ctx.writeAndFlush(new PingMessage());
             }
         }
     }
 }

項目源碼

  • UptimeClient
/**
 * Created by MySelf on 2019/8/27.
 */
public final class UptimeClient {

    static final String HOST = System.getProperty("host", "127.0.0.1");
    static final int PORT = Integer.parseInt(System.getProperty("port", "8080"));
    // 重新連接前睡眠5秒
    static final int RECONNECT_DELAY = Integer.parseInt(System.getProperty("reconnectDelay", "5"));
    // 當服務器在 10 秒內(nèi)不發(fā)送任何內(nèi)容時重新連接。
    private static final int READ_TIMEOUT = Integer.parseInt(System.getProperty("readTimeout", "10"));

    private static final UptimeClientHandler handler = new UptimeClientHandler();
    private static final Bootstrap bs = new Bootstrap();

    public static void main(String[] args) throws Exception {
        EventLoopGroup group = new NioEventLoopGroup();
        bs.group(group)
                .channel(NioSocketChannel.class)
                .remoteAddress(HOST,PORT)
                .handler(new ChannelInitializer<SocketChannel>() {
                    @Override
                    protected void initChannel(SocketChannel ch) throws Exception {
                        ch.pipeline().addLast(new IdleStateHandler(READ_TIMEOUT,0,0),handler);
                    }
                });
        bs.connect();
    }

    static void connect(){
        bs.connect().addListener(new ChannelFutureListener() {
            @Override
            public void operationComplete(ChannelFuture future) throws Exception {
                if (future.cause() != null){
                    handler.startTime = -1;
                    handler.println("Failed to connect:" + future.cause());
                }
            }
        });
    }

}
  • UptimeClientHandler
/**
 * Created by MySelf on 2019/8/27.
 */
@ChannelHandler.Sharable
public class UptimeClientHandler extends SimpleChannelInboundHandler<Object> {

    long startTime = -1;

    @Override
    protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {
        //Discard received data
    }

    @Override
    public void channelActive(ChannelHandlerContext ctx) throws Exception {
        if (startTime < 0){
            startTime = System.currentTimeMillis();
        }
        println("Connected to:" + ctx.channel().remoteAddress());
    }

    @Override
    public void channelInactive(ChannelHandlerContext ctx) throws Exception {
        println("Disconnected from: " + ctx.channel().remoteAddress());
    }

    @Override
    public void userEventTriggered(ChannelHandlerContext ctx, Object evt) throws Exception {
        if (!(evt instanceof IdleStateEvent)){
            return;
        }
        IdleStateEvent e = (IdleStateEvent)evt;
        if (e.state() == IdleState.READER_IDLE){
            // 連接正常秉氧,但是沒有讀信息眷昆,關(guān)閉連接
            println("Disconnecting due to no inbound traffic");
            ctx.close();
        }
    }

    @Override
    public void channelUnregistered(ChannelHandlerContext ctx) throws Exception {
        // 睡眠5秒
        println("Sleeping for:" + UptimeClient.RECONNECT_DELAY + 's');
        // 啟動線程重新連接
        ctx.channel().eventLoop().schedule(new Runnable() {
            @Override
            public void run() {
                println("Reconnecting to:" + UptimeClient.HOST + ":" + UptimeClient.PORT);
                UptimeClient.connect();
            }
        },UptimeClient.RECONNECT_DELAY, TimeUnit.SECONDS);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        cause.printStackTrace();
        ctx.close();
    }

    void println(String msg){
        if (startTime < 0){
            System.err.format("[SERVER IS DOWN] %s%n",msg);
        } else {
            System.err.format("[UPTIME: %5ds] %s%n",(System.currentTimeMillis() - startTime)/1000,msg);
        }
    }
}
  • UptimeServer
/**
 * Created by MySelf on 2019/8/27.
 */
public final class UptimeServer {

    private static final int PORT = Integer.parseInt(System.getProperty("port", "8080"));
    private static final UptimeServerHandler handler = new UptimeServerHandler();

    private UptimeServer(){}

    public static void main(String[] args) throws Exception {
        EventLoopGroup bossGroup = new NioEventLoopGroup();
        EventLoopGroup workerGroup = new NioEventLoopGroup();

        try {
            ServerBootstrap b = new ServerBootstrap();
            b.group(bossGroup,workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .handler(new LoggingHandler(LogLevel.INFO))
                    .childHandler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        protected void initChannel(SocketChannel ch) throws Exception {
                            ch.pipeline().addLast(handler);
                        }
                    });
            // Bind and start to accept incoming connections.
            ChannelFuture f = b.bind(PORT).sync();

            // Wait until the server socket is closed.
            // In this example, this does not happen, but you can do that to gracefully
            // shut down your server.
            f.channel().closeFuture().sync();
        }finally {
            workerGroup.shutdownGracefully();
            bossGroup.shutdownGracefully();
        }
    }
}
  • UptimeServerHandler
/**
 * Created by MySelf on 2019/8/27.
 */
@ChannelHandler.Sharable
public class UptimeServerHandler extends SimpleChannelInboundHandler<Object> {

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        // Close the connection when an exception is raised.
        cause.printStackTrace();
        ctx.close();
    }

    @Override
    protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {
        // discard
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市汁咏,隨后出現(xiàn)的幾起案子亚斋,更是在濱河造成了極大的恐慌,老刑警劉巖攘滩,帶你破解...
    沈念sama閱讀 216,470評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件帅刊,死亡現(xiàn)場離奇詭異,居然都是意外死亡漂问,警方通過查閱死者的電腦和手機赖瞒,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蚤假,“玉大人栏饮,你說我怎么就攤上這事×籽觯” “怎么了袍嬉?”我有些...
    開封第一講書人閱讀 162,577評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長灶平。 經(jīng)常有香客問我伺通,道長,這世上最難降的妖魔是什么逢享? 我笑而不...
    開封第一講書人閱讀 58,176評論 1 292
  • 正文 為了忘掉前任罐监,我火速辦了婚禮,結(jié)果婚禮上瞒爬,老公的妹妹穿的比我還像新娘笑诅。我一直安慰自己调缨,他們只是感情好,可當我...
    茶點故事閱讀 67,189評論 6 388
  • 文/花漫 我一把揭開白布吆你。 她就那樣靜靜地躺著弦叶,像睡著了一般。 火紅的嫁衣襯著肌膚如雪妇多。 梳的紋絲不亂的頭發(fā)上伤哺,一...
    開封第一講書人閱讀 51,155評論 1 299
  • 那天,我揣著相機與錄音者祖,去河邊找鬼立莉。 笑死,一個胖子當著我的面吹牛七问,可吹牛的內(nèi)容都是我干的蜓耻。 我是一名探鬼主播,決...
    沈念sama閱讀 40,041評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼械巡,長吁一口氣:“原來是場噩夢啊……” “哼刹淌!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起讥耗,我...
    開封第一講書人閱讀 38,903評論 0 274
  • 序言:老撾萬榮一對情侶失蹤有勾,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后古程,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蔼卡,經(jīng)...
    沈念sama閱讀 45,319評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,539評論 2 332
  • 正文 我和宋清朗相戀三年挣磨,在試婚紗的時候發(fā)現(xiàn)自己被綠了雇逞。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,703評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡茁裙,死狀恐怖塘砸,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情呜达,我是刑警寧澤谣蠢,帶...
    沈念sama閱讀 35,417評論 5 343
  • 正文 年R本政府宣布粟耻,位于F島的核電站查近,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏挤忙。R本人自食惡果不足惜霜威,卻給世界環(huán)境...
    茶點故事閱讀 41,013評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望册烈。 院中可真熱鬧戈泼,春花似錦婿禽、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,664評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至挽绩,卻和暖如春膛壹,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背唉堪。 一陣腳步聲響...
    開封第一講書人閱讀 32,818評論 1 269
  • 我被黑心中介騙來泰國打工模聋, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人唠亚。 一個月前我還...
    沈念sama閱讀 47,711評論 2 368
  • 正文 我出身青樓链方,卻偏偏與公主長得像,于是被迫代替她去往敵國和親灶搜。 傳聞我的和親對象是個殘疾皇子祟蚀,可洞房花燭夜當晚...
    茶點故事閱讀 44,601評論 2 353

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