netty入门篇(1)

上一篇 nio简介   下一篇 netty中级篇(2)

一、为什么选择Netty

Netty是最流行的框架之一、健壮性、功能、性能、可定制性和可扩展性在同类框架中首屈一指,因此被大规模使用,例如ROCKETMQ的NameSRV,例如Hadoop的Avro,例如Dubbo中的RPC通信等等。。

为什么选择Netty?

  • API简单;
  • 功能强大,预置了选多的编码功能,支持多种主流协议;
  • 定制能力强,通过ChannelHandler对通信框架进行灵活的扩展;
  • 性能强;
  • 成熟,修改已发现的JDK nio BUG
  • 社区活跃
  • 经过大规模的商业应用考验,质量得到验证。

二、使用Netty开发TimeServer

环境准备: pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>demo</groupId>
    <artifactId>netty</artifactId>
    <version>1.0-SNAPSHOT</version>
    <dependencies>
        <!-- https://mvnrepository.com/artifact/io.netty/netty-all -->
        <dependency>
            <groupId>io.netty</groupId>
            <artifactId>netty-all</artifactId>
            <version>4.1.5.Final</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.1</version>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

1. Netty TimeServer

 

import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
public class TimeServer {

    public void bind(int port) throws Exception {
        // 配置服务端的NIO线程组
        EventLoopGroup bossGroup = new NioEventLoopGroup();
        EventLoopGroup workerGroup = new NioEventLoopGroup();
        try {
            //Netty启动Nio服务端的辅助类
            ServerBootstrap b = new ServerBootstrap();
            b.group(bossGroup, workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .option(ChannelOption.SO_BACKLOG, 1024) //设置服务端tcp参数
                    .childHandler(new ChildChannelHandler());
            // 绑定端口,同步等待成功
            ChannelFuture f = b.bind(port).sync();

            // 进行阻塞,等待服务端监听端口关闭
            f.channel().closeFuture().sync();
        } finally {
            // 优雅退出,释放线程池资源
            System.out.println("服务器关闭...");
            bossGroup.shutdownGracefully();
            workerGroup.shutdownGracefully();
        }
    }

    private class ChildChannelHandler extends ChannelInitializer<SocketChannel> {
        @Override
        protected void initChannel(SocketChannel arg0) throws Exception {
            arg0.pipeline().addLast(new TimeServerHandler());
        }

    }

    /**
     * @param args
     * @throws Exception
     */
    public static void main(String[] args) throws Exception {
        int port = 8080;
        if (args != null && args.length > 0) {
            try {
                port = Integer.valueOf(args[0]);
            } catch (NumberFormatException e) {
                // 采用默认值
            }
        }
        new TimeServer().bind(port);
    }
}
  • EventLoopGroup Reactor线程组 2个
  • ServerBootstrap:Server端辅助工具
  • 设置channel: NioServerSocketChannel
  • option: 服务端tcp option设置,这里以backlog 1024为例..
  • 增加childHandler
  • f.channel().closeFuture().sync()表示进行阻塞,等待服务器端链路关闭之后main函数才退出

2. TimeServerHandler

 1 import io.netty.buffer.ByteBuf;
 2 import io.netty.buffer.Unpooled;
 3 import io.netty.channel.ChannelHandlerAdapter;
 4 import io.netty.channel.ChannelHandlerContext;
 5 import io.netty.channel.ChannelInboundHandlerAdapter;
 6 
 7 /**
 8  * @author lilinfeng
 9  * @version 1.0
10  * @date 2014年2月14日
11  */
12 public class TimeServerHandler extends ChannelInboundHandlerAdapter {
13 
14 
15     @Override
16     public void channelRead(ChannelHandlerContext ctx, Object msg)
17             throws Exception {
18         ByteBuf buf = (ByteBuf) msg;
19         byte[] req = new byte[buf.readableBytes()];
20         buf.readBytes(req);
21         String body = new String(req, "UTF-8");
22         System.out.println("The time server receive order : " + body);
23         String currentTime = "QUERY TIME ORDER".equalsIgnoreCase(body) ? new java.util.Date(
24                 System.currentTimeMillis()).toString() : "BAD ORDER";
25         ByteBuf resp = Unpooled.copiedBuffer(currentTime.getBytes());
26         ctx.write(resp);
27     }
28 
29     @Override
30     public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
31         ctx.flush();
32     }
33 
34     @Override
35     public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
36         ctx.close();
37     }
38 }

(1) 18行做类型转换,将msg转换为Netty的ByteBuf对象,这个对象比ByteBuffer更加强大和灵活。

(2) 19行到20行通过ByteBuf的readableBytes获取缓冲区可读的字节数,根据可读的字节数创建byte数组。将缓冲区的内容读取到byte数组中。

(3) 31行发现了flush方法,其作用是将消息发送队列中的消息写入到SocketChannel中发送给对方。从性能上考虑,为了防止频繁唤醒Selector进行消息发送,Netty的write方法不直接写入到SocketChannel中,调用write方法只会写入到缓冲数组中,调用flush方法,才会写入到SocketChannel中。

(4) 36行的close()是在发生异常后释放资源

总结: 就是比NIO舒服太多了.

3. Time Client

 1 import io.netty.bootstrap.Bootstrap;
 2 import io.netty.channel.ChannelFuture;
 3 import io.netty.channel.ChannelInitializer;
 4 import io.netty.channel.ChannelOption;
 5 import io.netty.channel.EventLoopGroup;
 6 import io.netty.channel.nio.NioEventLoopGroup;
 7 import io.netty.channel.socket.SocketChannel;
 8 import io.netty.channel.socket.nio.NioSocketChannel;
 9 
10 /**
11  * @author lilinfeng
12  * @version 1.0
13  * @date 2014年2月14日
14  */
15 public class TimeClient {
16 
17     public void connect(int port, String host) throws Exception {
18         // 配置客户端NIO线程组
19         EventLoopGroup group = new NioEventLoopGroup();
20         try {
21             Bootstrap b = new Bootstrap();
22             b.group(group).channel(NioSocketChannel.class)
23                     .option(ChannelOption.TCP_NODELAY, true)
24                     .handler(new ChannelInitializer<SocketChannel>() {
25                         @Override
26                         public void initChannel(SocketChannel ch)
27                                 throws Exception {
28                             ch.pipeline().addLast(new TimeClientHandler());
29                         }
30                     });
31 
32             // 发起异步连接操作
33             ChannelFuture f = b.connect(host, port).sync();
34 
35             // 当代客户端链路关闭
36             f.channel().closeFuture().sync();
37         } finally {
38             // 优雅退出,释放NIO线程组
39             group.shutdownGracefully();
40         }
41     }
42 
43     /**
44      * @param args
45      * @throws Exception
46      */
47     public static void main(String[] args) throws Exception {
48         int port = 8080;
49         if (args != null && args.length > 0) {
50             try {
51                 port = Integer.valueOf(args[0]);
52             } catch (NumberFormatException e) {
53                 // 采用默认值
54             }
55         }
56         new TimeClient().connect(port, "127.0.0.1");
57     }
58 }

(1) 19行创建客户端处理I/O读写的NioEventLoopGroup线程组,然后继续创建辅助类Bootstrap,并且对其配置,此处配置为 NioSocketChannel,然后为其添加Handler。

(2) 这里Handler直接使用匿名内部类

(3) 33行的connect发送异步连接请求,然后阻塞直到关闭。

4. TimeClientHandler

 1 import io.netty.buffer.ByteBuf;
 2 import io.netty.buffer.Unpooled;
 3 import io.netty.channel.ChannelHandlerAdapter;
 4 import io.netty.channel.ChannelHandlerContext;
 5 import io.netty.channel.ChannelInboundHandlerAdapter;
 6 
 7 import java.util.logging.Logger;
 8 
 9 /**
10  * @author lilinfeng
11  * @version 1.0
12  * @date 2014年2月14日
13  */
14 public class TimeClientHandler extends ChannelInboundHandlerAdapter {
15 
16     private static final Logger logger = Logger
17             .getLogger(TimeClientHandler.class.getName());
18 
19     private final ByteBuf firstMessage;
20 
21     /**
22      * Creates a client-side handler.
23      */
24     public TimeClientHandler() {
25         byte[] req = "QUERY TIME ORDER".getBytes();
26         firstMessage = Unpooled.buffer(req.length);
27         firstMessage.writeBytes(req);
28 
29     }
30 
31     @Override
32     public void channelActive(ChannelHandlerContext ctx) {
33         ctx.writeAndFlush(firstMessage);
34     }
35 
36     @Override
37     public void channelRead(ChannelHandlerContext ctx, Object msg)
38             throws Exception {
39         ByteBuf buf = (ByteBuf) msg;
40         byte[] req = new byte[buf.readableBytes()];
41         buf.readBytes(req);
42         String body = new String(req, "UTF-8");
43         System.out.println("Now is : " + body);
44     }
45 
46     @Override
47     public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
48         // 释放资源
49         logger.warning("Unexpected exception from downstream : "
50                 + cause.getMessage());
51         ctx.close();
52     }
53 }

 

这里重点关注3个方法: channelActive channelRead和exceptionCaught

(1) 当客户端和服务器端成功创建链路,调用channelActive方法,发送查询时间的指令给服务端,调用writeAndFlush方法发送数据。

(2) 39行开始调用channelRead,读取数据,49行处理异常时释放资源即可。

三、TCP 粘包/拆包问题的解决之道

1、TCP得粘包和拆包问题

  • TCP是一个流协议
  • TCP底层不了解业务数据含义,即不知道多少个字节算是业务上的一整体数据
  • 因此业务上认为,一个完整的包会被TCP拆分为多个包进行发送,也有可能将多个小的包封装成一个大包进行发送、

用下图进行描述,假设client发送了2个包,D1和D2,服务器端读到的数据是不确定的

存在4种可能:

server 分2次,分别读到D1,D2,完美巧合,没有粘包和拆包

server一次读到D1,D2,D1和D2粘在一起,称为粘包

server分2次,第一次读到D1和D2的部分内容,第二次读到了D2的剩余内容,拆包

server分2次,第一次读到D1的部分内容D1_1,第二次读到D1剩下的内容D1_2和完整的D2。

如果此时服务器端TCP接收的滑窗非常的小、而且数据包D1和D2都比较大,很有可能发生第5种可能性,服务器端多次才能将D1和D2接收完全,期间发生多次拆包...即上4种情况的多次组合...

下面我们来分析一下原因:

3个原因:

(1) 应用程序write写入的字节大于套接口(scoket)发送缓冲的大小;

(2) 进行MSS大小的TCP分段;

(3) 以太网帧的payload大于MTU进行IP分片

总结就是:不可避免...

解决思路:

(1) 定长数据,例如每个报文200bytes,不够空格补充...

(2) 在包围增加回车换行符或者其他的特殊字符进行分割,例如FTP协议

(3) 将消息分为消息头和消息体,消息头中包含消息总长度(或者消息体长度)content-length,通常的设计思路为消息头的第一个字段用int32来表示消息的总长度;

(4) 更复杂的应用层协议

2. 下面我们来模拟未考虑TCP粘包问题导致功能异常

修改上面的代码:

修改TimeServerHandler

 1 import io.netty.buffer.ByteBuf;
 2 import io.netty.buffer.Unpooled;
 3 import io.netty.channel.ChannelHandlerContext;
 4 import io.netty.channel.ChannelInboundHandlerAdapter;
 5 
 6 /**
 7  * @author lilinfeng
 8  * @version 1.0
 9  * @date 2014年2月14日
10  */
11 public class TimeServerHandler extends ChannelInboundHandlerAdapter {
12 
13     private int counter;
14 
15     @Override
16     public void channelRead(ChannelHandlerContext ctx, Object msg)
17             throws Exception {
18         ByteBuf buf = (ByteBuf) msg;
19         byte[] req = new byte[buf.readableBytes()];
20         buf.readBytes(req);
21         String body = new String(req, "UTF-8").substring(0, req.length
22                 - System.getProperty("line.separator").length());
23         System.out.println("The time server receive order : " + body
24                 + " ; the counter is : " + ++counter);
25         String currentTime = "QUERY TIME ORDER".equalsIgnoreCase(body) ? new java.util.Date(
26                 System.currentTimeMillis()).toString() : "BAD ORDER";
27         currentTime = currentTime + System.getProperty("line.separator");
28         ByteBuf resp = Unpooled.copiedBuffer(currentTime.getBytes());
29         ctx.writeAndFlush(resp);
30     }
31 
32     @Override
33     public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
34         ctx.close();
35     }
36 }

主要是增加了一个counter进行计数..

修改TimeClientHandler

import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;

import java.util.logging.Logger;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
public class TimeClientHandler extends ChannelInboundHandlerAdapter {

    private static final Logger logger = Logger
            .getLogger(TimeClientHandler.class.getName());

    private final ByteBuf firstMessage;

    /**
     * Creates a client-side handler.
     */
    public TimeClientHandler() {
        byte[] req = "QUERY TIME ORDER".getBytes();
        firstMessage = Unpooled.buffer(req.length);
        firstMessage.writeBytes(req);

    }

    @Override
    public void channelActive(ChannelHandlerContext ctx) {
        ctx.writeAndFlush(firstMessage);
    }

    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg)
            throws Exception {
        ByteBuf buf = (ByteBuf) msg;
        byte[] req = new byte[buf.readableBytes()];
        buf.readBytes(req);
        String body = new String(req, "UTF-8");
        System.out.println("Now is : " + body);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
        // 释放资源
        logger.warning("Unexpected exception from downstream : "
                + cause.getMessage());
        ctx.close();
    }
}

主要是进行100次连续的发送数据...

由于tcp粘包拆包有一定的随机性,所以每次的结果可能不同,其中一次结果大致上是:

Server端打印:

QUERY TIME ORDER
....
the counter is : 2

Client端打印:

Now is : Thu Dec 15 15:11:22 CST 2016
BAD ORDER
BAD ORDER
 ; the counter is : 1

结果表明:client发送了100条消息,但是server是按照2次接收,只返回2条应答,但是client上的counter为1表明只client也接收了一次,说明这2条也进行了粘包。

3. 解决TCP粘包的TimeServer

 

TimeServer

 1 import io.netty.bootstrap.ServerBootstrap;
 2 import io.netty.channel.ChannelFuture;
 3 import io.netty.channel.ChannelInitializer;
 4 import io.netty.channel.ChannelOption;
 5 import io.netty.channel.EventLoopGroup;
 6 import io.netty.channel.nio.NioEventLoopGroup;
 7 import io.netty.channel.socket.SocketChannel;
 8 import io.netty.channel.socket.nio.NioServerSocketChannel;
 9 import io.netty.handler.codec.LineBasedFrameDecoder;
10 import io.netty.handler.codec.string.StringDecoder;
11 
12 /**
13  * @author lilinfeng
14  * @version 1.0
15  * @date 2014年2月14日
16  */
17 public class TimeServer {
18 
19     public void bind(int port) throws Exception {
20         // 配置服务端的NIO线程组
21         EventLoopGroup bossGroup = new NioEventLoopGroup();
22         EventLoopGroup workerGroup = new NioEventLoopGroup();
23         try {
24             ServerBootstrap b = new ServerBootstrap();
25             b.group(bossGroup, workerGroup)
26                     .channel(NioServerSocketChannel.class)
27                     .option(ChannelOption.SO_BACKLOG, 1024)
28                     .childHandler(new ChildChannelHandler());
29             // 绑定端口,同步等待成功
30             ChannelFuture f = b.bind(port).sync();
31 
32             // 等待服务端监听端口关闭
33             f.channel().closeFuture().sync();
34         } finally {
35             // 优雅退出,释放线程池资源
36             bossGroup.shutdownGracefully();
37             workerGroup.shutdownGracefully();
38         }
39     }
40 
41     private class ChildChannelHandler extends ChannelInitializer<SocketChannel> {
42         @Override
43         protected void initChannel(SocketChannel arg0) throws Exception {
44             arg0.pipeline().addLast(new LineBasedFrameDecoder(1024));
45             arg0.pipeline().addLast(new StringDecoder());
46             arg0.pipeline().addLast(new TimeServerHandler());
47         }
48     }
49 
50     /**
51      * @param args
52      * @throws Exception
53      */
54     public static void main(String[] args) throws Exception {
55         int port = 8080;
56         if (args != null && args.length > 0) {
57             try {
58                 port = Integer.valueOf(args[0]);
59             } catch (NumberFormatException e) {
60                 // 采用默认值
61             }
62         }
63         new TimeServer().bind(port);
64     }
65 }

重点看44行,增加了2个解码器: LineBasedFrameDecoder和StringDecoder。

继续看TimeServerHandler

 1 import io.netty.buffer.ByteBuf;
 2 import io.netty.buffer.Unpooled;
 3 import io.netty.channel.ChannelHandlerContext;
 4 import io.netty.channel.ChannelInboundHandlerAdapter;
 5 
 6 /**
 7  * @author lilinfeng
 8  * @version 1.0
 9  * @date 2014年2月14日
10  */
11 public class TimeServerHandler extends ChannelInboundHandlerAdapter {
12 
13     private int counter;
14 
15     @Override
16     public void channelRead(ChannelHandlerContext ctx, Object msg)
17             throws Exception {
18         String body = (String) msg;
19         System.out.println("The time server receive order : " + body
20                 + " ; the counter is : " + ++counter);
21         String currentTime = "QUERY TIME ORDER".equalsIgnoreCase(body) ? new java.util.Date(
22                 System.currentTimeMillis()).toString() : "BAD ORDER";
23         currentTime = currentTime + System.getProperty("line.separator");
24         ByteBuf resp = Unpooled.copiedBuffer(currentTime.getBytes());
25         ctx.writeAndFlush(resp);
26     }
27 
28     @Override
29     public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
30         ctx.close();
31     }
32 }

看18行,直接获取之后不是ByteBuf,而直接是一个String对象,代码非常简洁。

TimeClient

 1 import io.netty.bootstrap.Bootstrap;
 2 import io.netty.channel.ChannelFuture;
 3 import io.netty.channel.ChannelInitializer;
 4 import io.netty.channel.ChannelOption;
 5 import io.netty.channel.EventLoopGroup;
 6 import io.netty.channel.nio.NioEventLoopGroup;
 7 import io.netty.channel.socket.SocketChannel;
 8 import io.netty.channel.socket.nio.NioSocketChannel;
 9 import io.netty.handler.codec.LineBasedFrameDecoder;
10 import io.netty.handler.codec.string.StringDecoder;
11 
12 /**
13  * @author lilinfeng
14  * @version 1.0
15  * @date 2014年2月14日
16  */
17 public class TimeClient {
18 
19     public void connect(int port, String host) throws Exception {
20         // 配置客户端NIO线程组
21         EventLoopGroup group = new NioEventLoopGroup();
22         try {
23             Bootstrap b = new Bootstrap();
24             b.group(group).channel(NioSocketChannel.class)
25                     .option(ChannelOption.TCP_NODELAY, true)
26                     .handler(new ChannelInitializer<SocketChannel>() {
27                         @Override
28                         public void initChannel(SocketChannel ch)
29                                 throws Exception {
30                             ch.pipeline().addLast(
31                                     new LineBasedFrameDecoder(1024));
32                             ch.pipeline().addLast(new StringDecoder());
33                             ch.pipeline().addLast(new TimeClientHandler());
34                         }
35                     });
36 
37             // 发起异步连接操作
38             ChannelFuture f = b.connect(host, port).sync();
39 
40             // 当代客户端链路关闭
41             f.channel().closeFuture().sync();
42         } finally {
43             // 优雅退出,释放NIO线程组
44             group.shutdownGracefully();
45         }
46     }
47 
48     /**
49      * @param args
50      * @throws Exception
51      */
52     public static void main(String[] args) throws Exception {
53         int port = 8080;
54         if (args != null && args.length > 0) {
55             try {
56                 port = Integer.valueOf(args[0]);
57             } catch (NumberFormatException e) {
58                 // 采用默认值
59             }
60         }
61         new TimeClient().connect(port, "127.0.0.1");
62     }
63 }

类似TimeServer增加了2个解码器

再看TimeClientHandler

import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;

import java.util.logging.Logger;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
public class TimeClientHandler extends ChannelInboundHandlerAdapter {

    private static final Logger logger = Logger
            .getLogger(TimeClientHandler.class.getName());

    private int counter;

    private byte[] req;

    /**
     * Creates a client-side handler.
     */
    public TimeClientHandler() {
        req = ("QUERY TIME ORDER" + System.getProperty("line.separator"))
                .getBytes();
    }

    @Override
    public void channelActive(ChannelHandlerContext ctx) {
        ByteBuf message = null;
        for (int i = 0; i < 100; i++) {
            message = Unpooled.buffer(req.length);
            message.writeBytes(req);
            ctx.writeAndFlush(message);
        }
    }

    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg)
            throws Exception {
        String body = (String) msg;
        System.out.println("Now is : " + body + " ; the counter is : "
                + ++counter);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
        // 释放资源
        logger.warning("Unexpected exception from downstream : "
                + cause.getMessage());
        ctx.close();
    }
}

直接运行发现完全符合我们需求

4. 分析LineBaseFrameDecoder和StringDecoder

LineBasedFrameDecoder的工作原理非常简单:

(1) 遍历ByteBuf中的可读字节,判断看是否有"\n"或者"\r\n",如果有,就以此为结束位置,从可读索引到结束位置区间的字节就组成了一行

(2) 是一个以换行符为结束标志的解码器,支持携带结束符或者不携带结束符2种方式,同时支持配置单行的最大长度。

(3) 超过单行最大长度直接抛异常

StringDecoder的非常简单:

(1) 将接收的对象转换为字符串

(2) 继续调用后面的Handler

因此:

LineBasedFrameDecoder和StringDecoder组合在一起就是行切换文件解码器。

四、分割符解码器的应用

使用DelimiterBasedFrameDecoder即可...

1. EohoServer

 1 import io.netty.bootstrap.ServerBootstrap;
 2 import io.netty.buffer.ByteBuf;
 3 import io.netty.buffer.Unpooled;
 4 import io.netty.channel.ChannelFuture;
 5 import io.netty.channel.ChannelInitializer;
 6 import io.netty.channel.ChannelOption;
 7 import io.netty.channel.EventLoopGroup;
 8 import io.netty.channel.nio.NioEventLoopGroup;
 9 import io.netty.channel.socket.SocketChannel;
10 import io.netty.channel.socket.nio.NioServerSocketChannel;
11 import io.netty.handler.codec.DelimiterBasedFrameDecoder;
12 import io.netty.handler.codec.string.StringDecoder;
13 import io.netty.handler.logging.LogLevel;
14 import io.netty.handler.logging.LoggingHandler;
15 
16 /**
17  * @author lilinfeng
18  * @version 1.0
19  * @date 2014年2月14日
20  */
21 public class EchoServer {
22     public void bind(int port) throws Exception {
23         // 配置服务端的NIO线程组
24         EventLoopGroup bossGroup = new NioEventLoopGroup();
25         EventLoopGroup workerGroup = new NioEventLoopGroup();
26         try {
27             ServerBootstrap b = new ServerBootstrap();
28             b.group(bossGroup, workerGroup)
29                     .channel(NioServerSocketChannel.class)
30                     .option(ChannelOption.SO_BACKLOG, 100)
31                     .handler(new LoggingHandler(LogLevel.INFO))
32                     .childHandler(new ChannelInitializer<SocketChannel>() {
33                         @Override
34                         public void initChannel(SocketChannel ch)
35                                 throws Exception {
36                             ByteBuf delimiter = Unpooled.copiedBuffer("$_"
37                                     .getBytes());
38                             ch.pipeline().addLast(
39                                     new DelimiterBasedFrameDecoder(1024,
40                                             delimiter));
41                             ch.pipeline().addLast(new StringDecoder());
42                             ch.pipeline().addLast(new EchoServerHandler());
43                         }
44                     });
45 
46             // 绑定端口,同步等待成功
47             ChannelFuture f = b.bind(port).sync();
48 
49             // 等待服务端监听端口关闭
50             f.channel().closeFuture().sync();
51         } finally {
52             // 优雅退出,释放线程池资源
53             bossGroup.shutdownGracefully();
54             workerGroup.shutdownGracefully();
55         }
56     }
57 
58     public static void main(String[] args) throws Exception {
59         int port = 8080;
60         if (args != null && args.length > 0) {
61             try {
62                 port = Integer.valueOf(args[0]);
63             } catch (NumberFormatException e) {
64                 // 采用默认值
65             }
66         }
67         new EchoServer().bind(port);
68     }
69 }

(1) 重点在于38行的DelimiterBasedFrameDecoder, 与上面的换行分割符类似,只是可以自定义特殊符号

(2) DelimiterBasedFrameDecoder有2个参数,一个为单行最大长度,一个为自定义符号对象

(3) 如果到达长度仍然没有查询到,就抛出TooLongFrameException异常

2. EchoServerHandler

import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandler.Sharable;
import io.netty.channel.ChannelHandlerAdapter;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
@Sharable
public class EchoServerHandler extends ChannelInboundHandlerAdapter {

    int counter = 0;

    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg)
            throws Exception {
        String body = (String) msg;
        System.out.println("This is " + ++counter + " times receive client : ["
                + body + "]");
        body += "$_";
        ByteBuf echo = Unpooled.copiedBuffer(body.getBytes());
        ctx.writeAndFlush(echo);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
        cause.printStackTrace();
        ctx.close();// 发生异常,关闭链路
    }
}

非常简单直接打印再write即可... 由此也可以看出netty框架比较干净的分离出来了业务逻辑代码。

3. Client端和ClientHandler基本类似

import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.codec.DelimiterBasedFrameDecoder;
import io.netty.handler.codec.string.StringDecoder;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
public class EchoClient {

    public void connect(int port, String host) throws Exception {
        // 配置客户端NIO线程组
        EventLoopGroup group = new NioEventLoopGroup();
        try {
            Bootstrap b = new Bootstrap();
            b.group(group).channel(NioSocketChannel.class)
                    .option(ChannelOption.TCP_NODELAY, true)
                    .handler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        public void initChannel(SocketChannel ch)
                                throws Exception {
                            ByteBuf delimiter = Unpooled.copiedBuffer("$_"
                                    .getBytes());
                            ch.pipeline().addLast(
                                    new DelimiterBasedFrameDecoder(1024,
                                            delimiter));
                            ch.pipeline().addLast(new StringDecoder());
                            ch.pipeline().addLast(new EchoClientHandler());
                        }
                    });

            // 发起异步连接操作
            ChannelFuture f = b.connect(host, port).sync();

            // 当代客户端链路关闭
            f.channel().closeFuture().sync();
        } finally {
            // 优雅退出,释放NIO线程组
            group.shutdownGracefully();
        }
    }

    /**
     * @param args
     * @throws Exception
     */
    public static void main(String[] args) throws Exception {
        int port = 8080;
        if (args != null && args.length > 0) {
            try {
                port = Integer.valueOf(args[0]);
            } catch (NumberFormatException e) {
                // 采用默认值
            }
        }
        new EchoClient().connect(port, "127.0.0.1");
    }
}
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerAdapter;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
public class EchoClientHandler extends ChannelInboundHandlerAdapter {

    private int counter;

    static final String ECHO_REQ = "Hi, Lilinfeng. Welcome to Netty.$_";

    /**
     * Creates a client-side handler.
     */
    public EchoClientHandler() {
    }

    @Override
    public void channelActive(ChannelHandlerContext ctx) {
        // ByteBuf buf = UnpooledByteBufAllocator.DEFAULT.buffer(ECHO_REQ
        // .getBytes().length);
        // buf.writeBytes(ECHO_REQ.getBytes());
        for (int i = 0; i < 10; i++) {
            ctx.writeAndFlush(Unpooled.copiedBuffer(ECHO_REQ.getBytes()));
        }
    }

    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg)
            throws Exception {
        System.out.println("This is " + ++counter + " times receive server : ["
                + msg + "]");
    }

    @Override
    public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
        ctx.flush();
    }

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

运行代码,符合预期..

五、定长解码器

1. 开发服务端

非常简单,直接上代码:

import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.FixedLengthFrameDecoder;
import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.logging.LogLevel;
import io.netty.handler.logging.LoggingHandler;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
public class EchoServer {
    public void bind(int port) throws Exception {
        // 配置服务端的NIO线程组
        EventLoopGroup bossGroup = new NioEventLoopGroup();
        EventLoopGroup workerGroup = new NioEventLoopGroup();
        try {
            ServerBootstrap b = new ServerBootstrap();
            b.group(bossGroup, workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .option(ChannelOption.SO_BACKLOG, 100)
                    .handler(new LoggingHandler(LogLevel.INFO))
                    .childHandler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        public void initChannel(SocketChannel ch)
                                throws Exception {
                            ch.pipeline().addLast(
                                    new FixedLengthFrameDecoder(20));
                            ch.pipeline().addLast(new StringDecoder());
                            ch.pipeline().addLast(new EchoServerHandler());
                        }
                    });

            // 绑定端口,同步等待成功
            ChannelFuture f = b.bind(port).sync();

            // 等待服务端监听端口关闭
            f.channel().closeFuture().sync();
        } finally {
            // 优雅退出,释放线程池资源
            bossGroup.shutdownGracefully();
            workerGroup.shutdownGracefully();
        }
    }

    public static void main(String[] args) throws Exception {
        int port = 8080;
        if (args != null && args.length > 0) {
            try {
                port = Integer.valueOf(args[0]);
            } catch (NumberFormatException e) {
                // 采用默认值
            }
        }
        new EchoServer().bind(port);
    }
}
import io.netty.channel.ChannelHandler.Sharable;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;

/**
 * @author lilinfeng
 * @version 1.0
 * @date 2014年2月14日
 */
@Sharable
public class EchoServerHandler extends ChannelInboundHandlerAdapter {

    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg)
            throws Exception {
        System.out.println("Receive client : [" + msg + "]");
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
        cause.printStackTrace();
        ctx.close();// 发生异常,关闭链路
    }
}

2. 使用telnet进行访问

(1) 我使用的是Xshell,直接命令

(2) telnet 127.0.0.1 8080

(3) 再随便输入字符,发现每20个字符,服务端显示一次,符合预期

 

posted @ 2016-12-15 16:12  carl_ysz  阅读(1656)  评论(2编辑  收藏  举报