当前位置: 首页 > news >正文

win7本机做网站工作室设计

win7本机做网站,工作室设计,博客seo怎么做,山东服务好的seo公司序列化的作用及自定义协议序列化的重要性大小对比效率对比自定义协议序列化数据结构自定义编码器自定义解码器安全性验证NettyClientNettyServerNettyClientTestHandlerNettyServerTestHandler结果上一章已经说了怎么解决沾包和拆包的问题#xff0c;但是这样离一个成熟的通信… 序列化的作用及自定义协议序列化的重要性大小对比效率对比自定义协议序列化数据结构自定义编码器自定义解码器安全性验证NettyClientNettyServerNettyClientTestHandlerNettyServerTestHandler结果上一章已经说了怎么解决沾包和拆包的问题但是这样离一个成熟的通信还是有一点距离我们还需要让服务端和客户端使用同一个语言来沟通要不然一个讲英文一个讲中文两个都听不懂岂不是很尴尬这种语言就叫协议。Netty自身就支持很多种协议比如Http、Websocket等等但如果用来作为自己的RPC框架通常会自定义协议所以这也是本文的重点 序列化的重要性 在说协议之前我们需要先知道什么是序列化序列化是干嘛的 我们要知道数据在传输的过程中是以0和1的形式传输的而把对象转化成二进制的过程就叫序列化将二进制转化为对象的过程就叫反序列化。 为什么要说这个很重要呢因为序列化和反序列化是需要耗时的而序列化后的字节大小也会影响到传输的效率所以选对一种高效的序列化方式是非常之重要的下面我们以JDK自带的序列化和我们常用的JSON序列化来做一个对比序列化后大小的对比、序列化效率的对比 大小对比 我们先准备一个实体类SerializeTestVO实现Serializable 接口 public class SerializeTestVO implements Serializable {private Integer id;private String name;private Integer age;private Integer sex;private Integer bodyWeight;private Integer height;private String school;//Set、get方法省略 }测试方法 public static void main(String[] args) throws IOException {// 普普通通的实体类SerializeTestVO serializeTestVO new SerializeTestVO();serializeTestVO.setAge(18);serializeTestVO.setBodyWeight(120);serializeTestVO.setHeight(180);serializeTestVO.setId(10000);serializeTestVO.setName(张三);serializeTestVO.setSchool(XXXXXXXXXXXX);// JDK序列化ByteArrayOutputStream byteArrayOutputStream new ByteArrayOutputStream();ObjectOutputStream objectOutputStream new ObjectOutputStream(byteArrayOutputStream);objectOutputStream.writeObject(serializeTestVO);objectOutputStream.flush();objectOutputStream.close();System.out.println(JDK 序列化大小: (byteArrayOutputStream.toByteArray().length));byteArrayOutputStream.close();//JSON序列化System.out.println(JSON 序列化大小: JSON.toJSONString(serializeTestVO).getBytes().length); }结果 可以看到序列化后大小相差了好几倍这也意味着传输效率的几倍 效率对比 实体类保持不变我们序列化300W次看看结果 public static void main(String[] args) throws IOException {SerializeTestVO serializeTestVO new SerializeTestVO();serializeTestVO.setAge(18);serializeTestVO.setBodyWeight(120);serializeTestVO.setHeight(180);serializeTestVO.setId(10000);serializeTestVO.setName(张三);serializeTestVO.setSchool(XXXXXXXXXXXX);long start System.currentTimeMillis();for (int i 0; i 3000000; i) {ByteArrayOutputStream byteArrayOutputStream new ByteArrayOutputStream();ObjectOutputStream objectOutputStream new ObjectOutputStream(byteArrayOutputStream);objectOutputStream.writeObject(serializeTestVO);objectOutputStream.flush();objectOutputStream.close();byte[] bytes byteArrayOutputStream.toByteArray();byteArrayOutputStream.close();}System.out.println(JDK 序列化耗时 (System.currentTimeMillis() - start));long start1 System.currentTimeMillis();for (int i 0; i 3000000; i) {byte[] bytes JSON.toJSONString(serializeTestVO).getBytes();}System.out.println(JSON 序列化耗时 (System.currentTimeMillis() - start1));}结果 几乎6倍的差距结合序列化后的大小综合来看选择一种好的序列化方式是多么的重要 自定义协议 其实到现在我们已经掌握了自定义协议里面最关键的几个点了序列化、数据结构、编解码器我们一个一个来 序列化 直接采用我们常用且熟悉的JSON序列化 数据结构 我们设置为消息头和消息体结构如下 消息头包含开始标志、时间戳、消息体长度 消息体包含通信凭证、消息ID、消息类型、消息 实体类如下 Data public class NettyMsg {private NettyMsgHead msgHeadnew NettyMsgHead();private NettyBody nettyBody;public NettyMsg(ServiceCodeEnum codeEnum, Object msg){this.nettyBodynew NettyBody(codeEnum, msg);} }Data public class NettyMsgHead {// 开始标识private short startSign (short) 0xFFFF;// 时间戳private final int timeStamp;public NettyMsgHead(){this.timeStamp(int)(DateUtil.current() / 1000);} }Data public class NettyBody {// 通信凭证private String token;// 消息IDprivate String msgId;// 消息类型private short msgType;// 消息 这里序列化采用JSON序列化// 所以这个msg可以是实体类的msg 两端通过消息类型来判断实体类类型private String msg;public NettyBody(){}public NettyBody(ServiceCodeEnum codeEnum,Object msg){this.token; // 鉴权使用this.msgId; // 拓展使用this.msgTypecodeEnum.getCode();this.msg JSON.toJSONString(msg);} } 消息类型枚举 JsonFormat(shape JsonFormat.Shape.OBJECT) public enum ServiceCodeEnum {TEST_TYPE((short) 0xFFF1, 测试);private final short code;private final String desc;ServiceCodeEnum(short code, String desc) {this.code code;this.desc desc;}public short getCode() {return code;}}自定义编码器 编码器的作用就是固定好我们的数据格式无需在每次发送数据的时候还需要去对数据进行格式编码 public class MyNettyEncoder extends MessageToByteEncoderNettyMsg {Overrideprotected void encode(ChannelHandlerContext channelHandlerContext, NettyMsg msg, ByteBuf out) throws Exception {// 写入开头的标志out.writeShort(msg.getMsgHead().getStartSign());// 写入秒时间戳out.writeInt(msg.getMsgHead().getTimeStamp());byte[] bytes JSON.toJSON(msg.getNettyBody()).toString().getBytes();// 写入消息长度out.writeInt(bytes.length);// 写入消息主体out.writeBytes(bytes);} }自定义解码器 解码器的第一个作用就是解决沾包和拆包的问题第二个作用就是对数据有效性的校验比如数据协议是否匹配、数据是否被篡改、数据加解密等等 所以我们直接继承LengthFieldBasedFrameDecoder类重写decode方法利用父类来解决沾包和拆包问题自定义来解决数据有效性问题 public class MyNettyDecoder extends LengthFieldBasedFrameDecoder {// 开始标记private final short HEAD_START (short) 0xFFFF;public MyNettyDecoder(int maxFrameLength, int lengthFieldOffset, int lengthFieldLength) {super(maxFrameLength, lengthFieldOffset, lengthFieldLength);}public MyNettyDecoder(int maxFrameLength, int lengthFieldOffset, int lengthFieldLength, int lengthAdjustment, int initialBytesToStrip) {super(maxFrameLength, lengthFieldOffset, lengthFieldLength, lengthAdjustment, initialBytesToStrip);}public MyNettyDecoder(int maxFrameLength, int lengthFieldOffset, int lengthFieldLength, int lengthAdjustment, int initialBytesToStrip, boolean failFast) {super(maxFrameLength, lengthFieldOffset, lengthFieldLength, lengthAdjustment, initialBytesToStrip, failFast);}public MyNettyDecoder(ByteOrder byteOrder, int maxFrameLength, int lengthFieldOffset, int lengthFieldLength, int lengthAdjustment, int initialBytesToStrip, boolean failFast) {super(byteOrder, maxFrameLength, lengthFieldOffset, lengthFieldLength, lengthAdjustment, initialBytesToStrip, failFast);}Overrideprotected Object decode(ChannelHandlerContext ctx, ByteBuf in) throws Exception {// 经过父解码器的处理 我们就不需要在考虑沾包和半包了// 当然想要自己处理沾包和半包问题也不是不可以ByteBuf decode (ByteBuf) super.decode(ctx, in);if (decode null) {return null;}// 开始标志校验 开始标志不匹配直接 过滤此条消息short startIndex decode.readShort();if (startIndex ! HEAD_START) {return null;}// 时间戳int timeIndex decode.readInt();// 消息体长度int lenOfBody decode.readInt();// 读取消息byte[] msgByte new byte[lenOfBody];decode.readBytes(msgByte);String msgContent new String(msgByte);// 将消息转成实体类 传递给下面的数据处理器return JSON.parseObject(msgContent, NettyBody.class);} }安全性 上述的协议里面我只预留了三种简单的校验一个是开始标识二是消息凭证三是时间戳实时上这太简单了下面我说几种可以加上去拓展的 消息整体加密消息头添加一个加密类型客户端和服务端都内置几种加解密手段在发送消息的时候随机一种加密方式对加密类型、消息长度以外的其他内容加密接收的时候再解密但是要注意加密后不能影响沾包和拆包的处理 消息体加密添加结束标识放入消息体和上述方式类似但是是对消息体中的内容再次加密可和上述方式结合形成二次加密 时间戳可以对长时间才接收到的消息拒收或者要求重发根据消息ID 加签和验签对具体的消息加签和验签防止篡改 凭证这个很熟悉了就比如登录凭证 复杂格式上述的数据格式还是过于简单实际可以整了更加复杂 验证 主体代码呢还是之前的我们改动几个地方 NettyClient 解码器是继承的LengthFieldBasedFrameDecoder所以参数也一样不懂的看一下上一篇 NettyServer NettyClientTestHandler 发送100次是为了验证沾包和拆包发送不同的开始标志是为了验证接收的时候是否有过滤无效数据 NettyServerTestHandler 有了编码器发送可以直接发送实体类有了解码器我们可以直接用实体类接收数据因为解码器里面往下传递的是过滤了消息头的实体类 结果 一共接收到了50条消息而且都是偶数消息说明无效消息被过滤了也没有沾包和拆包
http://www.w-s-a.com/news/886782/

相关文章:

  • 关注城市建设网站居众装饰
  • 网站建设的语言优化企业网站
  • 成都旅游网站建设规划女性门户资讯类网站织梦dedecms模板
  • 二手车为什么做网站网站建设合作合同范文
  • 网站建设维护和网页设计做网站都需要服务器吗
  • 成都网站设计报告书系统平台
  • 怎样进行网站推广wordpress微博图床
  • 做一个平台 网站服务器搭建网架公司股价
  • 链家在线网站是哪个公司做的一个虚拟主机做2个网站
  • 网站开发实训报告模板学校网站建设计划
  • 免费手机网站制作方法什么事网站开发
  • 我们的爱情网站制作阿里云wordpress配置
  • 电脑网站页面怎么调大小唐山网站建设技术外包
  • 科威网络做网站怎么样wordpress分页样式
  • 泰安公司网站建设自助建站程序
  • 网站建设工程设计图建网站怎样往网站传视频
  • 做网站月入企业网站建设运营
  • 网站建设中的ftp地址公众号微官网
  • 手机wap网站开发与设计app开发公司电话
  • 网站页脚代码大沥网站开发
  • 重庆网站制作公司 广州天成网络技术有限公司
  • 佛山网站改版wordpress 是否有后门
  • 如何承接网站建设外包wordpress产品布局
  • 洛阳建站洛阳市网站建设视觉设计专业
  • 婚恋网站建设分析网站建设硬件需求
  • 北京做网站电话wordpress如何换图片
  • 电影网站做cpa深圳信息网
  • 单县网站建设优化大师电脑版官网
  • 番禺区住房和建设局物业网站浦东新区网站设计
  • 外贸网站外包WordPress仿牌