类转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

posted @ 2021-01-19 15:43  阿飞飞啊飞  阅读(418)  评论(4编辑  收藏  举报