Netty的基本使用

Netty的基本使用

目录

[TOC]

引用

    implementation 'io.netty:netty-all:4.1.6.Final'

Netty 4最后一个release版本是4.1.28,但是亲测这个版本启动就会报异常...........所以降级回4.1.6

使用

TCP

服务端

    EventLoopGroup boss = new NioEventLoopGroup();
    EventLoopGroup worker = new NioEventLoopGroup();
            try {
        ServerBootstrap bootstrap = new ServerBootstrap();
        bootstrap.group(boss, worker);
        bootstrap.channel(NioServerSocketChannel.class);
        bootstrap.option(ChannelOption.TCP_NODELAY, true);
        bootstrap.childOption(ChannelOption.SO_KEEPALIVE, true);
        bootstrap.childHandler(new ChannelInitializer<SocketChannel>() {
            @Override
            protected void initChannel(SocketChannel socketChannel) throws Exception {
                ChannelPipeline p = socketChannel.pipeline();
                p.addLast("framedecoder", new LengthFieldBasedFrameDecoder(1024 * 1024 * 10, 0, 4, 0, 4));
                p.addLast(new NettyServerHandler());
            }
        });
        mChannelFuture = bootstrap.bind(mPort).sync();
        if (mChannelFuture.isSuccess()) {
            System.out.printf("start tcp server succeed,mPort : " + mPort);
        }
        mChannelFuture.channel().closeFuture().sync();
    } catch (Exception e) {
        System.out.printf("a exception happened while tcp server running");
        e.printStackTrace();
    } finally {
        boss.shutdownGracefully();
        worker.shutdownGracefully();
        mIsRunning = false;
    }

其中

mChannelFuture.channel().closeFuture().sync();

会阻塞线程,所以这段代码应该放在一个线程中.

Channel是连接服务端和客户端的通道.

ChannelPipeline p = socketChannel.pipeline();

ChannelPipeline是一个处理器容器,当有新数据接收时会调用容器中的处理器.

NettyServerHandler是处理最终接收到数据的地方

    private class NettyServerHandler extends ChannelInboundHandlerAdapter {

        @Override
        public void channelActive(ChannelHandlerContext ctx) throws Exception {
            initTransport(ctx.channel());
        }

        @Override
        public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
            WrapPacket packet = SocketUtils.decodeWrapPacketFromByteBufAfterSplit((ByteBuf) msg);
            String channelId = ctx.channel().id().asLongText();
            SubTransport transport = mChannelTransportMap.get(channelId);
            if (transport == null) {
                throw new RuntimeException("can not get transport form map , create a transport and put to map");
            }
            transport.onNewPacketReceived(packet);
        }

    }

LengthFieldBasedFrameDecoder是基于包长度的包解析器,用来拆包和处理半包.还有其他的自带解析器,比如分割,固定包长度等,不过个人偏向于喜欢这个,这个可以将资源利用最大化,而且没有字符不能使用的限制,如果是分割符的解析器,则会占用掉一些字符组合不能使用.当然这个也会翻车,当别人恶意攻击时,或者协议设计有问题时,一个错了,那就接下来所有都错了.错一步,满盘皆输,也是渗人.当然也可以自定义包解析器喽.

客户端

    EventLoopGroup eventLoopGroup = new NioEventLoopGroup();
            try {
        Bootstrap bootstrap = new Bootstrap();
        bootstrap.channel(NioSocketChannel.class);
        bootstrap.option(ChannelOption.SO_KEEPALIVE, true);
        bootstrap.group(eventLoopGroup);
        bootstrap.remoteAddress(mAddress, mPort);
        bootstrap.handler(new ChannelInitializer<SocketChannel>() {
            @Override
            protected void initChannel(SocketChannel socketChannel) throws Exception {
                socketChannel.pipeline().addLast("framedecoder", new LengthFieldBasedFrameDecoder(1024 * 1024 * 10, 0, 4, 0, 0));
                socketChannel.pipeline().addLast(new NettyClientHandler());
            }
        });
        mChannelFuture = bootstrap.connect(mAddress, mPort).sync();
        if (mChannelFuture.isSuccess()) {
            Log.d("connect to mAddress : " + mAddress + ", mPort : " + mPort + " succeed");
        }
        mChannelFuture.channel().closeFuture().sync();
    } catch (InterruptedException e) {
        e.printStackTrace();
        Log.d("can not connect to mAddress : " + mAddress + " ,mPort : " + mPort);
    } finally {
        eventLoopGroup.shutdownGracefully();
    }

同样也是阻塞的,和服务端差不多,便不再说明.

未完待续 ......

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容