java對(duì)象轉(zhuǎn)成byte數(shù)組的3種方法
java對(duì)象轉(zhuǎn)成byte數(shù)組,在使用netty進(jìn)行通信協(xié)議傳輸?shù)膱?chǎng)景中是非常常見的。比如,協(xié)議有一些定好的協(xié)議頭、classid,messageid等等信息,還有一個(gè)關(guān)鍵的內(nèi)容是payload。不同的協(xié)議內(nèi)容都會(huì)放到payload中,而這個(gè)payload往往就是一個(gè)byte數(shù)組。
那么,如何方便的將一個(gè)java對(duì)象構(gòu)造成一個(gè)byte數(shù)組呢?
1 bytebuf填充
我們以下面這個(gè)對(duì)象舉例:
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ù)編號(hào)
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è)上面的對(duì)象,調(diào)用其toByteArray方法,即可將這個(gè)對(duì)象轉(zhuǎn)成byte數(shù)組。
2 巧用json
我們都知道,字符串是可以轉(zhuǎn)成byte數(shù)組的。將一個(gè)對(duì)象轉(zhuǎn)成json字符串也很容易,直接使用fastjson就可以了。如果對(duì)fastjson使用有問(wèn)題的,可以看我的另一篇博客JSON.parseObject 和 JSON.toJSONString 實(shí)例
JSON.toJsonString(ugvData).getBytes()
3 反射的方式
第一種方法的缺點(diǎn)在于,每一個(gè)類都要這么寫一個(gè)toByteArray方法。如果類多了是非常麻煩的。有什么方便的方法嗎?當(dāng)然是有的,利用反射的方式(只會(huì)在第一次反射,后面會(huì)做本地緩存,所以性能開銷不大)。需要在一個(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ù)長(zhǎng)度。解碼時(shí)用,除了簡(jiǎn)單數(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è)類之后,使用也很簡(jiǎn)單,只需要如下所示,就可以把driveStartData轉(zhuǎn)成byte數(shù)組。
PayloadEncoder.getPayload(driveStartData)
4 總結(jié)
可能會(huì)有人問(wèn)了,上面三種,明顯第二種轉(zhuǎn)json最簡(jiǎn)單,為什么還要用另外兩種呢?
其實(shí),第一種和第三種可以歸為一類,都是把對(duì)象直接轉(zhuǎn)成byte數(shù)組,下一層做解析的話,可以一個(gè)一個(gè)元素??;
第二種情況是把對(duì)象的json字符串轉(zhuǎn)成byte數(shù)組,問(wèn)題就在于,json字符串最開頭是”{“,也就是轉(zhuǎn)成的byte數(shù)組的第一位是”{“對(duì)應(yīng)的數(shù)值
在使用中應(yīng)該根據(jù)情況來(lái),如果下一層做解析是直接取元素,對(duì)象少的話用第一種;對(duì)象多的話用第三種;
如果下一層做了排除掉json的一些格式的解析,就用第二種。
以上全部為本篇文章的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Java 字節(jié)數(shù)組(byte[])和整型(int)的相互轉(zhuǎn)換
- Java中字符串和byte數(shù)組之間的簡(jiǎn)單轉(zhuǎn)換方法
- 關(guān)于Java中byte[]?和?String互相轉(zhuǎn)換問(wèn)題
- java byte數(shù)組與16進(jìn)制間相互轉(zhuǎn)換的示例
- Java將byte[]轉(zhuǎn)圖片存儲(chǔ)到本地的案例
- Java如何把int類型轉(zhuǎn)換成byte
- java 文件和byte互轉(zhuǎn)的實(shí)例
- java byte與base64的互轉(zhuǎn)的實(shí)現(xiàn)示例
相關(guān)文章
Java tomcat中的類加載器和安全機(jī)制你了解嗎
這篇文章主要介紹了Tomcat 類加載器的實(shí)現(xiàn)方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下,希望能夠給你帶來(lái)幫助2021-09-09
java連接mysql數(shù)據(jù)庫(kù)亂碼的解決方法
這篇文章主要介紹通過(guò)java連接mysql數(shù)據(jù)庫(kù)的時(shí)候,頁(yè)面出現(xiàn)亂碼,這里簡(jiǎn)單分享下解決方法, 需要的朋友可以參考下2013-05-05
Spring中Controller和RestController的區(qū)別詳解
這篇文章主要介紹了Spring中Controller和RestController的區(qū)別詳解,@Controller是標(biāo)識(shí)一個(gè)Spring類是Spring MVC controller處理器,@Controller類中的方法可以直接通過(guò)返回String跳轉(zhuǎn)到j(luò)sp、ftl、html等模版頁(yè)面,需要的朋友可以參考下2023-09-09
解決Spring導(dǎo)出可以運(yùn)行的jar包問(wèn)題
最近需要解決Maven項(xiàng)目導(dǎo)入可執(zhí)行的jar包的問(wèn)題,如果項(xiàng)目不包含Spring,那么使用mvn assembly:assembly即可,這篇文章主要介紹了Spring導(dǎo)出可以運(yùn)行的jar包,需要的朋友可以參考下2023-03-03
記錄一個(gè)使用Spring?Data?JPA設(shè)置默認(rèn)值的問(wèn)題
這篇文章主要介紹了使用Spring?Data?JPA設(shè)置默認(rèn)值的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
Springboot集成graylog及配置過(guò)程解析
這篇文章主要介紹了Springboot集成graylog及配置過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-12-12
JVM運(yùn)行時(shí)數(shù)據(jù)區(qū)劃分原理詳解
這篇文章主要介紹了JVM運(yùn)行時(shí)數(shù)據(jù)區(qū)劃分原理詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05

