类转json的基类实现
类转json的基类实现
项目地址
实现原理
使用反射获取类的属性名和属性内容。具体原理可以自己查一下资料
对一个类调用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