java对象转成byte数组的3种方法
java对象转成byte数组,在使用netty进行通信协议传输的场景中是非常常见的。比如,协议有一些定好的协议头、classid,messageid等等信息,还有一个关键的内容是payload。不同的协议内容都会放到payload中,而这个payload往往就是一个byte数组。
那么,如何方便的将一个java对象构造成一个byte数组呢?
1bytebuf填充
我们以下面这个对象举例:
publicclassUgvDataimplementsSerializible{ privatestaticfinallongserialVersionUID=-219988432063763456L; //状态码 bytestatus; //当前GPS经度 floatlongitude; //当前GPS纬度 floatlatitude; //行驶速度单位是m/s,带一个小数点 floatspeed; //当前电量百分比 shortbatteryPercentage; //任务编号 longquest; publicbyte[]toByteArray(){ ByteBufbuf=Unpooled.buffer(32); buf.writeByte(this.getStatus()); buf.writeFloat(getLongitude()); buf.writeFloat(getLatitude()); buf.writeFloat(getSpeed()); buf.writeShort(getBatteryPercentage()); buf.writeLong(getQuest()); returnbuf.array(); } //省略getset }
那么只需要new出一个上面的对象,调用其toByteArray方法,即可将这个对象转成byte数组。
2巧用json
我们都知道,字符串是可以转成byte数组的。将一个对象转成json字符串也很容易,直接使用fastjson就可以了。如果对fastjson使用有问题的,可以看我的另一篇博客JSON.parseObject和JSON.toJSONString实例
JSON.toJsonString(ugvData).getBytes()
3反射的方式
第一种方法的缺点在于,每一个类都要这么写一个toByteArray方法。如果类多了是非常麻烦的。有什么方便的方法吗?当然是有的,利用反射的方式(只会在第一次反射,后面会做本地缓存,所以性能开销不大)。需要在一个文件夹下添加下面五个类
1.Codecable
importcom.fasterxml.jackson.annotation.JsonIgnore; importcom.google.common.collect.Lists; importlombok.Data; importjava.lang.reflect.Field; importjava.util.Collections; importjava.util.Comparator; importjava.util.List; @Data publicabstractclassCodecable{ publicstaticListresolveFileldWrapperList(Classclazz){ Field[]fields=clazz.getDeclaredFields(); List fieldWrapperList=Lists.newArrayList(); for(Fieldfield:fields){ CodecPropretycodecProprety=field.getAnnotation(CodecProprety.class); if(codecProprety==null){ continue; } FieldWrapperfw=newFieldWrapper(field,codecProprety); fieldWrapperList.add(fw); } Collections.sort(fieldWrapperList,newComparator (){ @Override publicintcompare(FieldWrappero1,FieldWrappero2){ returno1.getCodecProprety().order()-o2.getCodecProprety().order(); } }); returnfieldWrapperList; } @JsonIgnore publicabstractList getFieldWrapperList(); }
2.CodecProprety
importjava.lang.annotation.ElementType; importjava.lang.annotation.Retention; importjava.lang.annotation.RetentionPolicy; importjava.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME) @Target({ElementType.FIELD}) public@interfaceCodecProprety{ /** *属性顺序 *@return */ intorder(); /** *数据长度。解码时用,除了简单数据类型之外才起作用(如:String)。 *@return */ intlength()default0; }
3.FieldWrapper
importlombok.AllArgsConstructor; importlombok.Data; importjava.lang.reflect.Field; @Data @AllArgsConstructor publicclassFieldWrapper{ /** *上下行数据属性 */ privateFieldfield; /** *上下行数据属性上的注解 */ privateCodecPropretycodecProprety; }
4.PayloadDecoder
importio.netty.buffer.ByteBuf; importio.netty.buffer.Unpooled; importjava.lang.reflect.Field; importjava.lang.reflect.Method; importjava.nio.charset.Charset; importjava.util.List; publicclassPayloadDecoder{ publicstaticTresolve(byte[]src,Class clazz){ Tinstance=null; try{ instance=clazz.newInstance(); }catch(Exceptione){ thrownewRuntimeException("实例化类失败",e); } List fieldWrapperList=instance.getFieldWrapperList(); ByteBufbuffer=Unpooled.buffer().writeBytes(src); for(FieldWrapperfieldWrapper:fieldWrapperList){ fillData(fieldWrapper,instance,buffer); } returninstance; } privatestaticvoidfillData(FieldWrapperfieldWrapper,Objectinstance,ByteBufbuffer){ Fieldfield=fieldWrapper.getField(); field.setAccessible(true); StringtypeName=field.getType().getName(); try{ switch(typeName){ case"java.lang.Boolean": case"boolean": booleanb=buffer.readBoolean(); field.set(instance,b); break; case"java.lang.Character": case"char": CharSequencecharSequence=buffer.readCharSequence(fieldWrapper.getCodecProprety().length(),Charset.forName("UTF-8")); field.set(instance,charSequence); break; case"java.lang.Byte": case"byte": byteb1=buffer.readByte(); field.set(instance,b1); break; case"java.lang.Short": case"short": shortreadShort=buffer.readShort(); field.set(instance,readShort); break; case"java.lang.Integer": case"int": intreadInt=buffer.readInt(); field.set(instance,readInt); break; case"java.lang.Long": case"long": longl=buffer.readLong(); field.set(instance,l); break; case"java.lang.Float": case"float": floatreadFloat=buffer.readFloat(); field.set(instance,readFloat); break; case"java.lang.Double": case"double": doublereadDouble=buffer.readDouble(); field.set(instance,readDouble); break; case"java.lang.String": StringreadString=buffer.readCharSequence(fieldWrapper.getCodecProprety().length(),Charset.forName("UTF-8")).toString(); field.set(instance,readString); break; default: thrownewRuntimeException(typeName+"不支持,bug"); } }catch(Exceptione){ thrownewRuntimeException(typeName+"读取失败,field:"+field.getName(),e); } } }
5.PayloadEncoder
importio.netty.buffer.ByteBuf; importio.netty.buffer.Unpooled; importjava.lang.reflect.Field; importjava.lang.reflect.Method; importjava.nio.charset.Charset; importjava.util.List; publicclassPayloadEncoder{ publicstaticbyte[]getPayload(Tcommand){ List fieldWrapperList=command.getFieldWrapperList(); ByteBufbuffer=Unpooled.buffer(); fieldWrapperList.forEach(fieldWrapper->write2ByteBuf(fieldWrapper,command,buffer)); returnbuffer.array(); } /** *数据写入到ByteBuf * *@paramfieldWrapper *@paraminstance *@parambuffer */ privatestaticvoidwrite2ByteBuf(FieldWrapperfieldWrapper,Objectinstance,ByteBufbuffer){ Fieldfield=fieldWrapper.getField(); StringtypeName=field.getType().getName(); field.setAccessible(true); Objectvalue=null; try{ value=field.get(instance); }catch(IllegalAccessExceptione){ newRuntimeException("反射获取值失败,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: thrownewRuntimeException(typeName+"不支持,bug"); } } }
添加完上面五个类之后,使用也很简单,只需要如下所示,就可以把driveStartData转成byte数组。
PayloadEncoder.getPayload(driveStartData)
4总结
可能会有人问了,上面三种,明显第二种转json最简单,为什么还要用另外两种呢?
其实,第一种和第三种可以归为一类,都是把对象直接转成byte数组,下一层做解析的话,可以一个一个元素取;
第二种情况是把对象的json字符串转成byte数组,问题就在于,json字符串最开头是”{“,也就是转成的byte数组的第一位是”{“对应的数值
在使用中应该根据情况来,如果下一层做解析是直接取元素,对象少的话用第一种;对象多的话用第三种;
如果下一层做了排除掉json的一些格式的解析,就用第二种。
以上全部为本篇文章的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。