铁匠 铁匠
首页
收藏
java
架构之路
常用算法
  • Java
  • nginx
  • 系统运维
  • 系统安全
  • mysql
  • redis
参考文档
关于
链接
  • 分类
  • 标签
  • 归档

专注、不予评判地关注当下
首页
收藏
java
架构之路
常用算法
  • Java
  • nginx
  • 系统运维
  • 系统安全
  • mysql
  • redis
参考文档
关于
链接
  • 分类
  • 标签
  • 归档
  • java api 文档
  • 集合

  • 版本特性

  • jvm

  • 网络编程

    • Java IO
    • netty介绍
      • 优点
      • 沾包、拆包
      • 编解码
      • 使用
  • 并发编程

  • java
  • 网络编程
FengJianxin
2018-12-10
目录

netty介绍

# 优点

性能高、易用、java中几乎所有框架的网络通信都使用netty。

# 沾包、拆包

沾包、拆包问题的产生是因为TCP协议是基于数据流的协议,就像流水一样,没有严格的分界。发送数据的时候,可能有多条数据合并在一起,读取数据的时候也可能是多条数据,或者不完整的数据。

沾包:多条数据合并在一起

拆包:将读取的数据进行拆解,获得单独的一条完整数据

常见解决方案

  • 定长:预估数据长度,定义一个可以容纳所有数据的buffer长度,缺点是浪费带宽,适用于数据大小比较恒定的场景。
  • 换行符符:数据包以换行符结尾(\r or \r\n,例如netty中的LineBasedFrameDecoder)
  • 自定义分隔符:自定义一个分隔符作为数据结尾,缺点是可能会跟正常数据冲突。
  • 自定义消息格式:定义消息头和消息体,在消息头中包含消息长度信息。

# 编解码

网络通信的底层数据都是以字节传输,在接收到数据之后我们需要将字节码进行编码,得到业务数据,例如字符串,对象,httpReauest等。

netty已经内置了很多编解码器,例如:StringDecoder/StringEncoder

自定义编解码器,只需要实现MessageToMessageDecoder(解码)和MessageToMessageEncoder(编码)接口即可。

在很多RPC框架中都会自己实现一套编解码器,将数据反序列化成对象或者将对象序列化成字节。

# 使用

示例代码

jdk为了兼容各平台操作系统,对nio做了很多兼容处理,如果确定程序是在linux上运行,则可以直接使用epoll的实现。只需要将NioEventLoopGroup替换为EpollEventLoopGroup,并且将 NioServerSocketChannel.class 替换为 EpollServerSocketChannel.class 即可

import io.netty.bootstrap.ServerBootstrap;
import io.netty.buffer.Unpooled;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.codec.string.StringEncoder;
import io.netty.util.ReferenceCountUtil;

public class NettyServer {

    private final int port;

    public NettyServer(int port) {
        this.port = port;
    }

    public void start() throws Exception {
        EventLoopGroup group = new NioEventLoopGroup(5);
        EventLoopGroup worker = new NioEventLoopGroup(20);
        try {
            ServerBootstrap b = new ServerBootstrap();
            b.group(group, worker).channel(NioServerSocketChannel.class)
                    .childHandler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        protected void initChannel(SocketChannel sc) {
                            sc.pipeline().addLast(new StringDecoder());
                            sc.pipeline().addLast(new StringEncoder());
                            sc.pipeline().addLast(new EchoServerHandler());
                        }
                    });
            b.option(ChannelOption.SO_BACKLOG, 1024);
            b.childOption(ChannelOption.SO_KEEPALIVE, true);
            System.out.println("server start , port : " + port);
            ChannelFuture f = b.bind(port).sync();
            f.channel().closeFuture().sync();
        } finally {
            group.shutdownGracefully().sync();
        }
    }

    public static void main(String[] args) throws Exception {
        new NettyServer(6666).start();
    }

    static class EchoServerHandler extends ChannelInboundHandlerAdapter {

        @Override
        public void channelRead(ChannelHandlerContext ctx, Object msg) {
            try {
                String data = (String) msg;
                if (data == null || data.trim().length() == 0) {
                    return;
                }
                System.out.println("Server received: " + data);
                ctx.writeAndFlush(Unpooled.copiedBuffer(("received: " + data).getBytes()));
            } finally {
                ReferenceCountUtil.release(msg);
            }

        }

        @Override
        public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
            cause.printStackTrace();
            ctx.close();
        }
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69

依赖

<dependency>
    <groupId>io.netty</groupId>
    <artifactId>netty-all</artifactId>
    <version>4.1.31.Final</version>
</dependency>
1
2
3
4
5
#socket#io#netty
Java IO
并发编程三大特性

← Java IO 并发编程三大特性→

最近更新
01
策略模式
01-09
02
模板方法
01-06
03
观察者模式
01-06
更多文章>
Theme by Vdoing | Copyright © 2016-2023 铁匠 | 粤ICP备15021633号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式