欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java對象轉(zhuǎn)成byte數(shù)組的3種方法

 更新時(shí)間:2018年06月17日 14:26:17   作者:橙子wj  
這篇文章主要為大家詳細(xì)介紹了java對象轉(zhuǎn)成byte數(shù)組的3種方法,具有一定的參考價(jià)值,感興趣的朋友可以參考一下

java對象轉(zhuǎn)成byte數(shù)組,在使用netty進(jìn)行通信協(xié)議傳輸?shù)膱鼍爸惺欠浅3R姷?。比如,協(xié)議有一些定好的協(xié)議頭、classid,messageid等等信息,還有一個(gè)關(guān)鍵的內(nèi)容是payload。不同的協(xié)議內(nèi)容都會放到payload中,而這個(gè)payload往往就是一個(gè)byte數(shù)組。

那么,如何方便的將一個(gè)java對象構(gòu)造成一個(gè)byte數(shù)組呢?

1 bytebuf填充

我們以下面這個(gè)對象舉例:

public class UgvData implements Serializible{
 private static final long serialVersionUID = -219988432063763456L;

 //狀態(tài)碼
 byte status;
 //當(dāng)前GPS經(jīng)度
 float longitude;
 //當(dāng)前GPS緯度
 float latitude;
 //行駛速度 單位是 m/s,帶一個(gè)小數(shù)點(diǎn)
 float speed;
 //當(dāng)前電量百分比
 short batteryPercentage;
 //任務(wù)編號
 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出一個(gè)上面的對象,調(diào)用其toByteArray方法,即可將這個(gè)對象轉(zhuǎn)成byte數(shù)組。

2 巧用json

我們都知道,字符串是可以轉(zhuǎn)成byte數(shù)組的。將一個(gè)對象轉(zhuǎn)成json字符串也很容易,直接使用fastjson就可以了。如果對fastjson使用有問題的,可以看我的另一篇博客JSON.parseObject 和 JSON.toJSONString 實(shí)例

JSON.toJsonString(ugvData).getBytes()

3 反射的方式

第一種方法的缺點(diǎn)在于,每一個(gè)類都要這么寫一個(gè)toByteArray方法。如果類多了是非常麻煩的。有什么方便的方法嗎?當(dāng)然是有的,利用反射的方式(只會在第一次反射,后面會做本地緩存,所以性能開銷不大)。需要在一個(gè)文件夾下添加下面五個(gè)類

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();

 /**
  * 數(shù)據(jù)長度。解碼時(shí)用,除了簡單數(shù)據(jù)類型之外才起作用(如:String)。
  * @return
  */
 int length() default 0;
}

3.FieldWrapper

import lombok.AllArgsConstructor;
import lombok.Data;

import java.lang.reflect.Field;
@Data
@AllArgsConstructor
public class FieldWrapper {
 /**
  * 上下行數(shù)據(jù)屬性
  */
 private Field field;
 /**
  * 上下行數(shù)據(jù)屬性上的注解
  */
 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("實(shí)例化類失敗", 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();
 }

 /**
  * 數(shù)據(jù)寫入到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");
  }
 }
}

添加完上面五個(gè)類之后,使用也很簡單,只需要如下所示,就可以把driveStartData轉(zhuǎn)成byte數(shù)組。

PayloadEncoder.getPayload(driveStartData)

4 總結(jié)

可能會有人問了,上面三種,明顯第二種轉(zhuǎn)json最簡單,為什么還要用另外兩種呢?

其實(shí),第一種和第三種可以歸為一類,都是把對象直接轉(zhuǎn)成byte數(shù)組,下一層做解析的話,可以一個(gè)一個(gè)元素??;
第二種情況是把對象的json字符串轉(zhuǎn)成byte數(shù)組,問題就在于,json字符串最開頭是”{“,也就是轉(zhuǎn)成的byte數(shù)組的第一位是”{“對應(yīng)的數(shù)值

在使用中應(yīng)該根據(jù)情況來,如果下一層做解析是直接取元素,對象少的話用第一種;對象多的話用第三種;
如果下一層做了排除掉json的一些格式的解析,就用第二種。

以上全部為本篇文章的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • 詳解Java中switch的新特性

    詳解Java中switch的新特性

    這篇文章主要介紹了Java中switch的新特性,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05
  • 如何實(shí)現(xiàn)java Iterator迭代器功能

    如何實(shí)現(xiàn)java Iterator迭代器功能

    這篇文章主要介紹了如何實(shí)現(xiàn)java Iterator迭代器功能,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-01-01
  • Java中的Set集合不允許存儲重復(fù)元素的原理詳解

    Java中的Set集合不允許存儲重復(fù)元素的原理詳解

    這篇文章主要介紹了Java中的Set集合不允許存儲重復(fù)元素的原理詳解,我們之前使用Set集合的時(shí)候發(fā)現(xiàn),Set集合的特點(diǎn)是不允許存儲重復(fù)元素,這是為什么呢,下面我們一起來研究一下,需要的朋友可以參考下
    2023-09-09
  • Spring中@ConfigurationProperties的用法解析

    Spring中@ConfigurationProperties的用法解析

    這篇文章主要介紹了Spring中@ConfigurationProperties的用法解析,傳統(tǒng)的Spring一般都是基本xml配置的,后來spring3.0新增了許多java config的注解,特別是spring boot,基本都是清一色的java config,需要的朋友可以參考下
    2023-11-11
  • Java實(shí)現(xiàn)FTP文件上傳

    Java實(shí)現(xiàn)FTP文件上傳

    FTP 是File Transfer Protocol(文件傳輸協(xié)議)的英文簡稱,而中文簡稱為“文傳協(xié)議”。用于Internet上的控制文件的雙向傳輸。本文將講解如何用Java實(shí)現(xiàn)FTP文件上傳
    2021-05-05
  • MyBatis攔截器原理探究

    MyBatis攔截器原理探究

    MyBatis提供了一種插件(plugin)的功能,雖然叫做插件,但其實(shí)這是攔截器功能.這篇文章主要介紹了MyBatis攔截器原理探究,需要的朋友可以參考下
    2018-02-02
  • Java父類繼承中的static和final用法

    Java父類繼承中的static和final用法

    這篇文章主要介紹了Java父類繼承中的static和final用法說明,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10
  • java提供的4種函數(shù)式接口

    java提供的4種函數(shù)式接口

    這篇文章主要介紹了java提供的4種函數(shù)式接口,函數(shù)接口是只有一個(gè)抽象方法的接口,用作?Lambda?表達(dá)式的類型并有且只有一個(gè)抽象方法的接口,下文詳細(xì)內(nèi)容,需要的小伙伴可以參考一下
    2022-03-03
  • nexus安裝及配置圖文教程

    nexus安裝及配置圖文教程

    Nexus 是Maven倉庫管理器,通過nexus可以搭建maven倉庫,同時(shí)nexus還提供強(qiáng)大的倉庫管理功能,構(gòu)件搜索功能等,文中有非常詳細(xì)的圖文介紹,對小伙伴們很有幫助,需要的朋友可以參考下
    2021-05-05
  • Java實(shí)現(xiàn)經(jīng)典游戲復(fù)雜迷宮

    Java實(shí)現(xiàn)經(jīng)典游戲復(fù)雜迷宮

    這篇文章主要介紹了如何利用java語言實(shí)現(xiàn)經(jīng)典《復(fù)雜迷宮》游戲,文中采用了swing技術(shù)進(jìn)行了界面化處理,感興趣的小伙伴可以動(dòng)手試一試
    2022-02-02

最新評論