Netty4 学习笔记之一:客户端与服务端通信 demo

ゝ一纸荒年。 2022-06-09 08:10 543阅读 0赞

前言

因为以前在项目中使用过Mina框架,感受到了该框架的强大之处。于是在业余时间也学习了一下Netty。因为Netty的主要版本是Netty3和Netty4(Netty5已经被取消了),所以我就直接学习Netty4。在本文中演示的就是Netty的一个简单demo。

开发准备

Netty4的官方网站是:http://netty.io/ 。
本文使用的是Netty4.1。
由于Netty4.1版本需要JDK1.7,在使用JDK1.6时,会有一些bug,所以推荐使用JDK1.7。

因为Netty和Mina都是Trustin Lee的作品,所以在很多方面都十分相似,他们线程模型也是基本一致,采用了Reactors in threads模型,即Main Reactor + Sub Reactors的模式。

所以在开发上,很多都可以相互借鉴。

服务端

首先完成Netty服务端相关开发,和Mina一样。
主要由两个部分组成: 配置服务端的基本信息以及实现业务逻辑处理。
如果需要,还有filter过滤器。

服务端代码

  1. import io.netty.bootstrap.ServerBootstrap;
  2. import io.netty.channel.ChannelFuture;
  3. import io.netty.channel.EventLoopGroup;
  4. import io.netty.channel.nio.NioEventLoopGroup;
  5. import io.netty.channel.socket.nio.NioServerSocketChannel;
  6. /** * * Title: NettyServer * Description: Netty服务端 * Version:1.0.0 * @author Administrator * @date 2017-8-31 */
  7. public class NettyServer {
  8. private static final int port = 6789; //设置服务端端口
  9. private static EventLoopGroup group = new NioEventLoopGroup(); // 通过nio方式来接收连接和处理连接
  10. private static ServerBootstrap b = new ServerBootstrap();
  11. /** * Netty创建全部都是实现自AbstractBootstrap。 * 客户端的是Bootstrap,服务端的则是 ServerBootstrap。 **/
  12. public static void main(String[] args) throws InterruptedException {
  13. try {
  14. b.group(group);
  15. b.channel(NioServerSocketChannel.class);
  16. b.childHandler(new NettyServerFilter()); //设置过滤器
  17. // 服务器绑定端口监听
  18. ChannelFuture f = b.bind(port).sync();
  19. System.out.println("服务端启动成功...");
  20. // 监听服务器关闭监听
  21. f.channel().closeFuture().sync();
  22. } finally {
  23. group.shutdownGracefully(); 关闭EventLoopGroup,释放掉所有资源包括创建的线程
  24. }
  25. }
  26. }

服务端业务逻辑

  1. import java.net.InetAddress;
  2. import java.util.Date;
  3. import io.netty.channel.ChannelHandlerContext;
  4. import io.netty.channel.SimpleChannelInboundHandler;
  5. /** * * Title: HelloServerHandler * Description: 服务端业务逻辑 * Version:1.0.0 * @author Administrator * @date 2017-8-31 */
  6. public class NettyServerHandler extends SimpleChannelInboundHandler<String> {
  7. /* * 收到消息时,返回信息 */
  8. @Override
  9. protected void channelRead0(ChannelHandlerContext ctx, String msg)
  10. throws Exception {
  11. // 收到消息直接打印输出
  12. System.out.println("服务端接受的消息 : " + msg);
  13. if("quit".equals(msg)){
  14. //服务端断开的条件
  15. ctx.close();
  16. }
  17. Date date=new Date();
  18. // 返回客户端消息
  19. ctx.writeAndFlush(date+"\n");
  20. }
  21. /* * 建立连接时,返回消息 */
  22. @Override
  23. public void channelActive(ChannelHandlerContext ctx) throws Exception {
  24. System.out.println("连接的客户端地址:" + ctx.channel().remoteAddress());
  25. ctx.writeAndFlush("客户端"+ InetAddress.getLocalHost().getHostName() + "成功与服务端建立连接! \n");
  26. super.channelActive(ctx);
  27. }
  28. }

服务端过滤器

  1. import io.netty.channel.ChannelInitializer;
  2. import io.netty.channel.ChannelPipeline;
  3. import io.netty.channel.socket.SocketChannel;
  4. import io.netty.handler.codec.DelimiterBasedFrameDecoder;
  5. import io.netty.handler.codec.Delimiters;
  6. import io.netty.handler.codec.string.StringDecoder;
  7. import io.netty.handler.codec.string.StringEncoder;
  8. /** * * Title: HelloServerInitializer * Description: Netty 服务端过滤器 * Version:1.0.0 * @author Administrator * @date 2017-8-31 */
  9. public class NettyServerFilter extends ChannelInitializer<SocketChannel> {
  10. @Override
  11. protected void initChannel(SocketChannel ch) throws Exception {
  12. ChannelPipeline ph = ch.pipeline();
  13. // 以("\n")为结尾分割的 解码器
  14. ph.addLast("framer", new DelimiterBasedFrameDecoder(8192, Delimiters.lineDelimiter()));
  15. // 解码和编码,应和客户端一致
  16. ph.addLast("decoder", new StringDecoder());
  17. ph.addLast("encoder", new StringEncoder());
  18. ph.addLast("handler", new NettyServerHandler());// 服务端业务逻辑
  19. }
  20. }

客户端

客户端的主要工作是
1,连接到服务端
2,向服务端发送数据数据
3,处理服务端返回的数据
4,关闭连接
而且客户端相关代码也和服务端类似。

客户端

  1. import io.netty.bootstrap.Bootstrap;
  2. import io.netty.channel.Channel;
  3. import io.netty.channel.EventLoopGroup;
  4. import io.netty.channel.nio.NioEventLoopGroup;
  5. import io.netty.channel.socket.nio.NioSocketChannel;
  6. import java.io.IOException;
  7. /** * * Title: NettyClient * Description: Netty客户端 * Version:1.0.0 * @author Administrator * @date 2017-8-31 */
  8. public class NettyClient {
  9. public static String host = "127.0.0.1"; //ip地址
  10. public static int port = 6789; //端口
  11. /// 通过nio方式来接收连接和处理连接
  12. private static EventLoopGroup group = new NioEventLoopGroup();
  13. private static Bootstrap b = new Bootstrap();
  14. private static Channel ch;
  15. /** * Netty创建全部都是实现自AbstractBootstrap。 * 客户端的是Bootstrap,服务端的则是 ServerBootstrap。 **/
  16. public static void main(String[] args) throws InterruptedException, IOException {
  17. System.out.println("客户端成功启动...");
  18. b.group(group);
  19. b.channel(NioSocketChannel.class);
  20. b.handler(new NettyClientFilter());
  21. // 连接服务端
  22. ch = b.connect(host, port).sync().channel();
  23. star();
  24. }
  25. public static void star() throws IOException{
  26. String str="Hello Netty";
  27. ch.writeAndFlush(str+ "\r\n");
  28. System.out.println("客户端发送数据:"+str);
  29. }
  30. }

客户端业务逻辑实现

  1. import io.netty.channel.ChannelHandlerContext;
  2. import io.netty.channel.SimpleChannelInboundHandler;
  3. /** * * Title: NettyClientHandler * Description: 客户端业务逻辑实现 * Version:1.0.0 * @author Administrator * @date 2017-8-31 */
  4. public class NettyClientHandler extends SimpleChannelInboundHandler<String> {
  5. @Override
  6. protected void channelRead0(ChannelHandlerContext ctx, String msg) throws Exception {
  7. System.out.println("客户端接受的消息: " + msg);
  8. }
  9. //
  10. @Override
  11. public void channelActive(ChannelHandlerContext ctx) throws Exception {
  12. System.out.println("正在连接... ");
  13. super.channelActive(ctx);
  14. }
  15. @Override
  16. public void channelInactive(ChannelHandlerContext ctx) throws Exception {
  17. System.out.println("连接关闭! ");
  18. super.channelInactive(ctx);
  19. }
  20. }

客户端过滤器

  1. import io.netty.channel.ChannelInitializer;
  2. import io.netty.channel.ChannelPipeline;
  3. import io.netty.channel.socket.SocketChannel;
  4. import io.netty.handler.codec.DelimiterBasedFrameDecoder;
  5. import io.netty.handler.codec.Delimiters;
  6. import io.netty.handler.codec.string.StringDecoder;
  7. import io.netty.handler.codec.string.StringEncoder;
  8. /** * * Title: NettyClientFilter * Description: Netty客户端 过滤器 * Version:1.0.0 * @author Administrator * @date 2017-8-31 */
  9. public class NettyClientFilter extends ChannelInitializer<SocketChannel> {
  10. @Override
  11. protected void initChannel(SocketChannel ch) throws Exception {
  12. ChannelPipeline ph = ch.pipeline();
  13. /* * 解码和编码,应和服务端一致 * */
  14. ph.addLast("framer", new DelimiterBasedFrameDecoder(8192, Delimiters.lineDelimiter()));
  15. ph.addLast("decoder", new StringDecoder());
  16. ph.addLast("encoder", new StringEncoder());
  17. ph.addLast("handler", new NettyClientHandler()); //客户端的逻辑
  18. }
  19. }

效果实现图

服务端

这里写图片描述

客户端

这里写图片描述

demo教程到这里就结束了,如果有什么问题,欢迎提出!

该项目我放在github上了,有兴趣的可以看看!https://github.com/xuwujing/Netty

发表评论

表情:
评论列表 (有 0 条评论,543人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Netty实现客户服务通信

    实现一个客户端与服务端通信的程序,可以使用socket网络编程来实现,而Netty作为一个封装了JDK的NIO通讯的异步事件驱动的网络应用框架,也同样可以实现。 1.创建M