添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
相关文章推荐
爱运动的热带鱼  ·  webstorm - Get ...·  1 年前    · 
有腹肌的熊猫  ·  jquery 判断select ...·  2 年前    · 
更新时间:2018年06月17日 14:26:17   作者:橙子wj
这篇文章主要为大家详细介绍了java对象转成byte数组的3种方法,具有一定的参考价值,感兴趣的朋友可以参考一下

java对象转成byte数组,在使用netty进行通信协议传输的场景中是非常常见的。比如,协议有一些定好的协议头、classid,messageid等等信息,还有一个关键的内容是payload。不同的协议内容都会放到payload中,而这个payload往往就是一个byte数组。

那么,如何方便的将一个java对象构造成一个byte数组呢?

1 bytebuf填充

我们以下面这个对象举例:

public class UgvData implements Serializible{ private static final long serialVersionUID = -219988432063763456L; //状态码 byte status; //当前GPS经度 float longitude; //当前GPS纬度 float latitude; //行驶速度 单位是 m/s,带一个小数点 float speed; //当前电量百分比 short batteryPercentage; //任务编号 long quest; public byte[] toByteArray() { ByteBuf buf = Unpooled.buffer(32); buf.writeByte(this.getStatus()); buf.writeFloat(getLongitude()); buf.writeFloat(getLatitude()); buf.writeFloat(getSpeed()); buf.writeShort(getBatteryPercentage()); buf.writeLong(getQuest()); return buf.array(); //省略get set

那么只需要new出一个上面的对象,调用其toByteArray方法,即可将这个对象转成byte数组。

2 巧用json

我们都知道,字符串是可以转成byte数组的。将一个对象转成json字符串也很容易,直接使用fastjson就可以了。如果对fastjson使用有问题的,可以看我的另一篇博客 JSON.parseObject 和 JSON.toJSONString 实例

JSON.toJsonString(ugvData).getBytes()

3 反射的方式

第一种方法的缺点在于,每一个类都要这么写一个toByteArray方法。如果类多了是非常麻烦的。有什么方便的方法吗?当然是有的,利用反射的方式(只会在第一次反射,后面会做本地缓存,所以性能开销不大)。需要在一个文件夹下添加下面五个类

1.Codecable

import com.fasterxml.jackson.annotation.JsonIgnore; import com.google.common.collect.Lists; import lombok.Data; import java.lang.reflect.Field; import java.util.Collections; import java.util.Comparator; import java.util.List; @Data public abstract class Codecable { public static List<FieldWrapper> resolveFileldWrapperList(Class clazz){ Field[] fields = clazz.getDeclaredFields(); List<FieldWrapper> fieldWrapperList = Lists.newArrayList(); for (Field field : fields) { CodecProprety codecProprety = field.getAnnotation(CodecProprety.class); if (codecProprety == null) { continue; FieldWrapper fw = new FieldWrapper(field, codecProprety); fieldWrapperList.add(fw); Collections.sort(fieldWrapperList, new Comparator<FieldWrapper>() { @Override public int compare(FieldWrapper o1, FieldWrapper o2) { return o1.getCodecProprety().order() - o2.getCodecProprety().order(); return fieldWrapperList; @JsonIgnore public abstract List<FieldWrapper> getFieldWrapperList();

2.CodecProprety

import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME) @Target({ElementType.FIELD}) public @interface CodecProprety { * 属性顺序 * @return int order(); * 数据长度。解码时用,除了简单数据类型之外才起作用(如:String)。 * @return int length() default 0;

3.FieldWrapper

import lombok.AllArgsConstructor; import lombok.Data; import java.lang.reflect.Field; @Data @AllArgsConstructor public class FieldWrapper { * 上下行数据属性 private Field field; * 上下行数据属性上的注解 private CodecProprety codecProprety;

4.PayloadDecoder

import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled; import java.lang.reflect.Field; import java.lang.reflect.Method; import java.nio.charset.Charset; import java.util.List; public class PayloadDecoder { public static <T extends Codecable> T resolve(byte[] src, Class<T> clazz) { T instance = null; try { instance = clazz.newInstance(); } catch (Exception e) { throw new RuntimeException("实例化类失败", e); List<FieldWrapper> fieldWrapperList = instance.getFieldWrapperList(); ByteBuf buffer = Unpooled.buffer().writeBytes(src); for (FieldWrapper fieldWrapper : fieldWrapperList) { fillData(fieldWrapper, instance, buffer); return instance; private static void fillData(FieldWrapper fieldWrapper, Object instance, ByteBuf buffer) { Field field = fieldWrapper.getField(); field.setAccessible(true); String typeName = field.getType().getName(); try { switch (typeName) { case "java.lang.Boolean": case "boolean": boolean b = buffer.readBoolean(); field.set(instance, b); break; case "java.lang.Character": case "char": CharSequence charSequence = buffer.readCharSequence(fieldWrapper.getCodecProprety().length(), Charset.forName("UTF-8")); field.set(instance, charSequence); break; case "java.lang.Byte": case "byte": byte b1 = buffer.readByte(); field.set(instance, b1); break; case "java.lang.Short": case "short": short readShort = buffer.readShort(); field.set(instance, readShort); break; case "java.lang.Integer": case "int": int readInt = buffer.readInt(); field.set(instance, readInt); break; case "java.lang.Long": case "long": long l = buffer.readLong(); field.set(instance, l); break; case "java.lang.Float": case "float": float readFloat = buffer.readFloat(); field.set(instance, readFloat); break; case "java.lang.Double": case "double": double readDouble = buffer.readDouble(); field.set(instance, readDouble); break; case "java.lang.String": String readString = buffer.readCharSequence(fieldWrapper.getCodecProprety().length(), Charset.forName("UTF-8")).toString(); field.set(instance, readString); break; default: throw new RuntimeException(typeName + "不支持,bug"); } catch (Exception e) { throw new RuntimeException(typeName + "读取失败,field:" + field.getName(), e);

5.PayloadEncoder

import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled; import java.lang.reflect.Field; import java.lang.reflect.Method; import java.nio.charset.Charset; import java.util.List; public class PayloadEncoder { public static <T extends Codecable> byte[] getPayload(T command) { List<FieldWrapper> fieldWrapperList = command.getFieldWrapperList(); ByteBuf buffer = Unpooled.buffer(); fieldWrapperList.forEach(fieldWrapper -> write2ByteBuf(fieldWrapper, command, buffer)); return buffer.array(); * 数据写入到ByteBuf * @param fieldWrapper * @param instance * @param buffer private static void write2ByteBuf(FieldWrapper fieldWrapper, Object instance, ByteBuf buffer) { Field field = fieldWrapper.getField(); String typeName = field.getType().getName(); field.setAccessible(true); Object value = null; try { value = field.get(instance); } catch (IllegalAccessException e) { new RuntimeException("反射获取值失败,filed:" + field.getName(), e); switch (typeName) { case "java.lang.Boolean": case "boolean": buffer.writeBoolean((Boolean) value); break; case "java.lang.Character": case "char": buffer.writeCharSequence((CharSequence) value, Charset.forName("UTF-8")); break; case "java.lang.Byte": case "byte": buffer.writeByte((byte) value); break; case "java.lang.Short": case "short": buffer.writeShort((short) value); break; case "java.lang.Integer": case "int": buffer.writeInt((int) value); break; case "java.lang.Long": case "long": buffer.writeLong((long) value); break; case "java.lang.Float": case "float": buffer.writeFloat((float) value); break; case "java.lang.Double": case "double": buffer.writeDouble((double) value); break; case "java.lang.String": buffer.writeCharSequence((CharSequence) value, Charset.forName("UTF-8")); break; default: throw new RuntimeException(typeName + "不支持,bug");

添加完上面五个类之后,使用也很简单,只需要如下所示,就可以把driveStartData转成byte数组。

PayloadEncoder.getPayload(driveStartData)

可能会有人问了,上面三种,明显第二种转json最简单,为什么还要用另外两种呢?

其实,第一种和第三种可以归为一类,都是把对象直接转成byte数组,下一层做解析的话,可以一个一个元素取;
第二种情况是把对象的json字符串转成byte数组,问题就在于,json字符串最开头是”{“,也就是转成的byte数组的第一位是”{“对应的数值

在使用中应该根据情况来,如果下一层做解析是直接取元素,对象少的话用第一种;对象多的话用第三种;
如果下一层做了排除掉json的一些格式的解析,就用第二种。

以上全部为本篇文章的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
  • SpringBoot使用自动配置xxxAutoConfiguration

    SpringBoot使用自动配置xxxAutoConfiguration

    这篇文章介绍了SpringBoot自动配置xxxAutoConfiguration的使用方法,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-12-12
  • Spring事务注解@Transactional失效的八种场景分析

    Spring事务注解@Transactional失效的八种场景分析

    最近在开发采用Spring框架的项目中,使用了@Transactional注解,但发现事务注解失效了,所以这篇文章主要给大家介绍了关于Spring事务注解@Transactional失效的八种场景,需要的朋友可以参考下
    2021-05-05
  • 全面汇总SpringBoot和SpringClould常用注解

    全面汇总SpringBoot和SpringClould常用注解

    Java注解是附加在代码中的一些元信息,用于一些工具在编译、运行时进行解析和使用,起到说明、配置的功能,这篇文章就带你来了解一下
    2021-08-08
  • java HashMap,TreeMap与LinkedHashMap的详解

    java HashMap,TreeMap与LinkedHashMap的详解

    这篇文章主要介绍了 java HashMap,TreeMap与LinkedHashMap的详解的相关资料,这里提供实例代码,帮助大家学习理解 这部分的内容,需要的朋友可以参考下
    2016-11-11
  • Spring Cloud 动态刷新配置信息教程详解

    Spring Cloud 动态刷新配置信息教程详解

    这篇文章主要介绍了Spring Cloud 动态刷新配置信息的教程,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-06-06
  • SpringBoot集成使用Redis及搭建过程

    SpringBoot集成使用Redis及搭建过程

    jackson-json 工具提供了 javabean 与 json 之 间的转换能力,可以将 pojo 实例序列化成 json 格式存储在 redis 中,也可以将 json 格式的数据转换成 pojo 实例,本文给大家介绍SpringBoot集成使用Redis及搭建过程,感兴趣的朋友一起看看吧
    2022-01-01
  • Springboot如何读取resources下的json配置文件

    Springboot如何读取resources下的json配置文件

    这篇文章主要介绍了Springboot如何读取resources下的json配置文件问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • Java 实战范例之进销存管理系统的实现

    Java 实战范例之进销存管理系统的实现

    读万卷书不如行万里路,只学书上的理论是远远不够的,只有在实战中才能获得能力的提升,本篇文章手把手带你用java+vue+Springboot+ssm+mysql+maven+redis实现一个前后端分离的进销存管理系统,大家可以在过程中查缺补漏,提升水平
    2021-11-11
  • java实现纸牌游戏之小猫钓鱼算法

    java实现纸牌游戏之小猫钓鱼算法

    这篇文章主要为大家详细介绍了java实现纸牌游戏之小猫钓鱼算法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • SpringBoot解析yml全流程详解

    SpringBoot解析yml全流程详解

    本文主要介绍了SpringBoot解析yml全流程详解,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03