第一种:自定义规则

比如说我们自己设定$_结尾的数据为一个整体。

看主要代码,大体不变,就多了几行代码。具体先看我上一篇的代码。这里只做修改

server端

  1. b.childHandler(new ChannelInitializer<SocketChannel>() {
  2. @Override
  3. protected void initChannel(SocketChannel ch) throws Exception {
  4. ByteBuf byteBuf = Unpooled.copiedBuffer("$_".getBytes());
  5. ch.pipeline().addLast(new DelimiterBasedFrameDecoder(1024,byteBuf));//自定义分割标识
  6. ch.pipeline().addLast(new StringDecoder()); //解码类,类似过滤器,处理类接收到数据就不用转到buf再转string了
  7. ch.pipeline().addLast(new MyServerHandler());
  8.  
  9. }
  10. });

服务端处理类

  1. @Override
  2. public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception{
  3. //读取客户端发来的信息
  4. String m = (String) msg;
  5. System.out.println("client:"+m);
  6. //向客户端写信息
  7. String name="你好客户端:这是服务端返回的信息!$_";
  8. ctx.writeAndFlush(Unpooled.copiedBuffer(name.getBytes()));
  9. }

client端

  1. package com.lzh.netty;
  2.  
  3. import io.netty.bootstrap.Bootstrap;
  4. import io.netty.buffer.ByteBuf;
  5. import io.netty.buffer.Unpooled;
  6. import io.netty.channel.ChannelFuture;
  7. import io.netty.channel.ChannelInitializer;
  8. import io.netty.channel.ChannelOption;
  9. import io.netty.channel.EventLoopGroup;
  10. import io.netty.channel.nio.NioEventLoopGroup;
  11. import io.netty.channel.socket.SocketChannel;
  12. import io.netty.channel.socket.nio.NioSocketChannel;
  13. import io.netty.handler.codec.DelimiterBasedFrameDecoder;
  14. import io.netty.handler.codec.string.StringDecoder;
  15.  
  16. /**
  17. * Created by 敲代码的卡卡罗特
  18. * on 2018/8/12 21:46.
  19. */
  20. public class Client {
  21.  
  22. public static void main(String[] arg){
  23. /**
  24. * 如果你只指定了一个EventLoopGroup,
  25. * 那他就会即作为一个‘boss’线程,
  26. * 也会作为一个‘workder’线程,
  27. * 尽管客户端不需要使用到‘boss’线程。
  28. */
  29. Bootstrap b = new Bootstrap(); // (1)
  30. EventLoopGroup workerGroup = new NioEventLoopGroup();
  31. b.group(workerGroup); // (2)
  32. b.channel(NioSocketChannel.class); // (3)
  33. b.option(ChannelOption.SO_KEEPALIVE, true); // (4)
  34. b.handler(new ChannelInitializer<SocketChannel>() {
  35. @Override
  36. public void initChannel(SocketChannel ch) throws Exception {
  37. ByteBuf byteBuf = Unpooled.copiedBuffer("$_".getBytes());
  38. ch.pipeline().addLast(new DelimiterBasedFrameDecoder(1024,byteBuf));//自定义分割标识
  39. ch.pipeline().addLast(new StringDecoder()); //解码类
  40. ch.pipeline().addLast(new MyClientHandler());
  41. }
  42. });
  43.  
  44. try {
  45. ChannelFuture f = b.connect("127.0.0.1", 8888).sync();
  46. //向服务端发送信息
  47. f.channel().writeAndFlush(Unpooled.copiedBuffer("你好$_哈哈$_".getBytes()));
  48.  
  49. f.channel().closeFuture().sync();
  50. } catch (InterruptedException e) {
  51. e.printStackTrace();
  52. } finally {
  53. workerGroup.shutdownGracefully();
  54. }
  55. }
  56. }

客户端处理类

  1. package com.lzh.netty;
  2.  
  3. import io.netty.buffer.ByteBuf;
  4. import io.netty.channel.ChannelHandlerAdapter;
  5. import io.netty.channel.ChannelHandlerContext;
  6. import io.netty.util.CharsetUtil;
  7. import io.netty.util.ReferenceCountUtil;
  8.  
  9. /**
  10. * Created by 敲代码的卡卡罗特
  11. * on 2018/8/12 21:49.
  12. */
  13. public class MyClientHandler extends ChannelHandlerAdapter {
  14.  
  15. @Override
  16. public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception{
  17. try {
  18. //读取服务端发来的信息
  19. String m = (String) msg; // ByteBuf是netty提供的
  20. System.out.println("client:"+m);
  21. } catch (Exception e) {
  22. e.printStackTrace();
  23. } finally {
  24. //当没有写操作的时候要把msg给清空。如果有写操作,就不用清空,因为写操作会自动把msg清空。这是netty的特性。
  25. ReferenceCountUtil.release(msg);
  26. }
  27.  
  28. }
  29.  
  30. @Override
  31. public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
  32. cause.printStackTrace();
  33. ctx.close();
  34. }
  35. }

第二种:定长的分隔符

其实很简单,就是加上一条限制。

  1. ch.pipeline().addLast(new FixedLengthFrameDecoder(5)); //定长的分隔符

netty的拆包和粘包的更多相关文章

  1. netty的解码器和粘包拆包

    Tcp是一个流的协议,一个完整的包可能会被Tcp拆成多个包进行发送,也可能把一个小的包封装成一个大的数据包发送,这就是所谓的粘包和拆包问题 粘包.拆包出现的原因: 在流传输中出现,UDP不会出现粘包, ...

  2. Netty系列(四)TCP拆包和粘包

    Netty系列(四)TCP拆包和粘包 一.拆包和粘包问题 (1) 一个小的Socket Buffer问题 在基于流的传输里比如 TCP/IP,接收到的数据会先被存储到一个 socket 接收缓冲里.不 ...

  3. Netty处理TCP拆包、粘包

    Netty实践(二):TCP拆包.粘包问题-学海无涯 心境无限-51CTO博客 http://blog.51cto.com/zhangfengzhe/1890577 2017-01-09 21:56: ...

  4. netty 拆包和粘包 (三)

    在tcp编程底层都有拆包和粘包的机制   拆包 当发送数据量过大时数据量会分多次发送 以前面helloWord代码为例 package com.liqiang.nettyTest2; public c ...

  5. netty权威指南学习笔记三——TCP粘包/拆包之粘包现象

    TCP是个流协议,流没有一定界限.TCP底层不了解业务,他会根据TCP缓冲区的实际情况进行包划分,在业务上,一个业务完整的包,可能会被TCP底层拆分为多个包进行发送,也可能多个小包组合成一个大的数据包 ...

  6. 深入了解Netty【八】TCP拆包、粘包和解决方案

    1.TCP协议传输过程 TCP协议是面向流的协议,是流式的,没有业务上的分段,只会根据当前套接字缓冲区的情况进行拆包或者粘包: 发送端的字节流都会先传入缓冲区,再通过网络传入到接收端的缓冲区中,最终由 ...

  7. 【转】Netty之解决TCP粘包拆包(自定义协议)

    1.什么是粘包/拆包 一般所谓的TCP粘包是在一次接收数据不能完全地体现一个完整的消息数据.TCP通讯为何存在粘包呢?主要原因是TCP是以流的方式来处理数据,再加上网络上MTU的往往小于在应用处理的消 ...

  8. Netty之解决TCP粘包拆包(自定义协议)

    1.什么是粘包/拆包 一般所谓的TCP粘包是在一次接收数据不能完全地体现一个完整的消息数据.TCP通讯为何存在粘包呢?主要原因是TCP是以流的方式来处理数据,再加上网络上MTU的往往小于在应用处理的消 ...

  9. netty的解码器与粘包和拆包

    tcp是一个“流”的协议,一个完整的包可能会被TCP拆分成多个包进行发送,也可能把小的封装成一个大的数据包发送,这就是所谓的TCP粘包和拆包问题. 假设客户端分别发送数据包D1和D2给服务端,由于服务 ...

随机推荐

  1. Luogu4726 【模板】多项式指数函数(NTT+多项式求逆)

    https://www.cnblogs.com/HocRiser/p/8207295.html 安利! #include<iostream> #include<cstdio> ...

  2. 清北澡堂 Day2 上午 一些比较重要的关于数论的知识整理

    1.算数基本定理: 对于任意的大于1的正整数N,N一定能够分解成有限个质数的乘积,即 其中P1<P2<...<Pk,a1,a2,...,ak>=1; 证: 存在性: 若存在最小 ...

  3. ecplise properties文件 中文转码

    1.安装插件 2.重开ecplise 3.在项目的乱码文件如jeesite.properties右键 openwith propertiesEditor 就可以看到中文了 输入 proedit 安装完 ...

  4. 【刷题】BZOJ 1413 [ZJOI2009]取石子游戏

    Description 在研究过Nim游戏及各种变种之后,Orez又发现了一种全新的取石子游戏,这个游戏是这样的: 有n堆石子,将这n堆石子摆成一排.游戏由两个人进行,两人轮流操作,每次操作者都可以从 ...

  5. FlatList

    FlatList 之前使用的组件是ListView,当时要添加一个下拉刷新,上拉加载的功能,所以对ListView做了一些封装,但是后来看官方文档,不建议再使用ListView,因为效率问题,做过An ...

  6. module_loader.py

    # few functions that make it possible to import functions # from jupyter notebooks as from modules; ...

  7. Squid代理服务部署

    构建Squid代理服务器1.配置IP地址 2.编译安装Squid软件[root@localhost ~]# tar -zxvf squid-3.4.6.tar.gz -C /usr/src/[root ...

  8. How To Install WildFly as a Service on Linux

    Installing WildFly as a service on Linux has multiple advantages like automatic start on system boot ...

  9. C# DateTimePicker控件获取他的年,月,日,时,分,秒

    CustomFormat属性设置为: yyyy-MM-dd HH:mm:ss 记住还要修改一个属性值,DateFormat属性 可选项改为Custom,默认是Long

  10. A1110. Complete Binary Tree

    Given a tree, you are supposed to tell if it is a complete binary tree. Input Specification: Each in ...