類轉json的基類實現


類轉json的基類實現

項目地址

github地址

實現原理

使用反射獲取類的屬性名和屬性內容。具體原理可以自己查一下資料

對一個類調用getClass().getDeclaredFields()可以獲取許多信息。

通過上述方法獲取到的是一個Field類型的數組,這個類中有多少屬性就有多少field,包括用private修飾的屬性。

調用field.getName即可獲得string類型的屬性名,如下

for(Field field : fields){
  String name = field.getName();
}

有了這個屬性名稱還不夠,還需要屬性的內容,使用field.get()方法,get中放入父類

Field[] fields = this.getClass().getDeclaredFields();
//遍歷
for(Field field : fields){
  Object object = field.get(this);

這樣子就可以直接獲取到屬性內容,然后這里只需要創建一個JsonObject,把屬性名和屬性內容一一填入即可,最多再加一個判斷內容是不是為null即可。

一個雛形方法就出來了,到這也可以簡單的輸出一個JsonObject的對象了。

升級版

直接使用這個方法,會發現有些參數的值怎么有點不協調,是類型加hash的形式,這樣可不太好。如果是自定義類型那也就算了,當一些基礎類型數組形式下輸出也是這樣,那可就不太好了。不過如果是list的話卻沒有問題

先調用getType方法判斷下數組傳入時的類型,然后根據這些類型,創建對應的list,將他們填入list中,將得到的list放到JsonObject中,當然也可以直接用一個JsonObject去封裝一個出來,然后放進去,但是看起來怪怪的。

如下是我自己測了一下各基本類型的type,

case "[Z": //boolean
case "[B"://byte
case "[C"://char
case "[D"://double
case "[F"://float
case "[J"://long
case "[S"://short
case "[I"://int

如果是大寫的基礎類型是繼承自Object的,這種比較好統一處理,type類型是由“[L”開頭,寫個if判斷下就好了。把數組轉為list的方法如下

/**
* 數組轉成list
* */
private List arrayToList(Object object, @NotNull ObjectType type){
  List a = new ArrayList();
  switch (type){
    case INT:
      for(int i:(int[]) object){
        a.add(i);
      }
      break;
    case BOOLEAN:
      for(boolean i:(boolean[]) object){
        a.add(i);
      }
      break;
    case LONG:
      for(long i:(long[]) object){
        a.add(i);
      }
      break;
    case FLOAT:
      for(float i:(float[]) object){
        a.add(i);
      }
      break;
    case DOUBLE:
      for(Double i:(double[]) object){
        a.add(i);
      }
      break;
    case BYTE:
      for(byte i:(byte[]) object){
        a.add(i);
      }
      break;
    case CHAR:
      for(char i:(char[]) object){
        a.add(i);
      }
      break;
    case SHORT:
      for(short i:(short[]) object){
        a.add(i);
      }
      break;
    case OBJECT:
      for(Object i:(Object[]) object){
        a.add(i);
      }
      break;
    default:
      a = null;
      break;
  }
  return a;
}

我自己寫了個枚舉,當他為Object時,表示是繼承自Object的類型處理起來就很方便了。直接把list類型的參數存入JsonObject中也不會出現hash值。

最終版

如果屬性是一個自定義類怎么辦呢,如果自定義類中還有一個自定義類呢。

這問題其實很簡單,如果屬性是自定義類怎么辦,那肯定是拆他,怎么拆,他爹怎么被拆的就怎么拆他。只需要遞歸就可以解決。

但是如果自定義類是數組形式怎么辦,之前只是列出了基礎類型的type,如果是java自己的類,肯定是java開頭的,所以只需如下判斷

if(type.startsWith("java")){
    return ObjectType.NOTLIST;
}else if(type.startsWith("[L")){
    //繼承了Object的數組
    if(type.startsWith("[Ljava"))
        return ObjectType.OBJECT;
    return ObjectType.NOTBASELIST;
}

"[L"則是數組類型的前綴,"[Ljava"則是表示java自帶類型,所以剔除這兩個之外就是自定義類了,如果你的類的包名正好也是java開頭的,那還有個辦法,把你自己的包名填入這里如“[Ljava.util.”,把自己篩出來。或者直接用改成“[Ljava.util.”排除所有java.util下的包也可。具體看自己操作。

反射遇上混淆

當反射遇到混淆的時候發現怎么回事,沒辦法用了。因為反射是根據類名來獲取,名稱什么都沒了,那就什么都沒獲取,只需要加一下混淆規則即可

-keep class com.example.androidstudy.ToJson.ToJson{*;}
-keep  class * extends com.example.androidstudy.ToJson.ToJson{
    *;
}
-keepnames class * extends com.example.androidstudy.ToJson.ToJson{
   *;
}

這個項目的實現使用android實現的,如果是java的話就自己寫寫實現看看效果。具體使用可以詳見github里的readme


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM