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个字符,服务端显示一次,符合预期