Gson的反射解析机制详解(2)

来源:互联网 发布:黑马程序员全套视频 编辑:程序博客网 时间:2024/06/05 09:25

在上一篇博客中笼统的介绍了Gson中解析json的整体流程,但是具体细节并没有说明,本篇就简单的梳理一下:
1)怎么利用反射来创建自定义的JavaBean?
2)怎么给JavaBen的变量类来赋值?
3)集合类对象怎么创建?

通过阅读Gson源码可得出以下的结论:
1)先 获取type获取JavaBean的java.lang.reflect.Constructor,构造器为默认构造器
2)通过Constructor的newInstance()来创建一个type类型的JavaBean对象。
3)循环遍历json中的键值对,获取key和value;并且获取key对应的JavaBean中的Field
4)将value通过Filed的set(JavaBean,value)方法,将value赋值给Javaben对应的Field中去

实例代码如下:

        Constructor<?> c = Person.class.getDeclaredConstructor();        Person  person=(Personn) c.newInstance();        //获取Person中的变量        Field[] fields = Person.class.getDeclaredFields();        for (Field field : fields) {            field.setAccessible(true);            //获取变量名                String name =  field.getName();          if (name.equals("name"){                          field.set(person, "凌云");          } else if (name.equals("sex")) {            field.set(person, "屌丝男");          } else if(name.equals("job")){            field.set(person,"IT monkey");          }          }              return person

上面的例子程序可以说是Gson中对json解析的简单雏形,其实这些结论都是Gson对于Java反射知识的基本应用,只不过在用的时候结合泛型做了有效的封装来进行通用。下面就简单的梳理一下Gson是怎么来进行处理的。
1)json串键值对key/value中key绑定JavaBean中对应的变量。(本节假设JavaBean中便令名没有用到注解), 在Gson中创建了一个BoundField的类,该类就是负责json中key与JavaBean的变量名进行映射绑定:

   //   private final Map<String, BoundField> boundFields;    //该类在ReflectiveTypeAdapterFactory中定义  static abstract class BoundField {    //json的key或者JavaBean中的变量名    final String name;    ...... protected BoundField(String name, boolean serialized, boolean deserialized) {      this.name = name;      this.serialized = serialized;      this.deserialized = deserialized;    }

在ReflectiveTypeAdapterFactory类中通过getBoundFields方法来对boundFields(详见Gson的反射解析机制详解(1)这个map进行初始化,主要是循环遍历JavaBean中的变量,然后把每个变量封装成一个BoundField,放入map中.该map的key对应的是JavaBean的变量名,value对应的就是BoundField。而BoundField还负责对其绑定的JavaBean变量或者json中key进行读取解析,然后赋值给该JavaBean变量,通过Field的set(JavaBean,value)来完成(这部分详见Gson的反射解析机制详解(1) )。.方法基本的骨架跟文章开头类似。

2)步骤1)中提到,最终是通过Field.setValue(JavaBean,value)来完成当前变量的赋值操作,方法参数中value我们知道是通过读取json来获取,而第一个参数JavaBean是怎么创建的?答案是通过反射创建Gson 的TypeToken 中T代表的对象,也就是JavaBean对象。具体的就是用Constructor.newInstance();见文章开头实例方法:但在Gson中是通过ConstructorConstructor这个类来完成这一工作的:该类提供了newDefaultConstructor来获取JavaBean中的默认构造器,并通过该构造器创建一个JavaBean对象:

 private <T> ObjectConstructor<T> newDefaultConstructor(Class<? super T> rawType) {      //此方法返回具有指定参数列表构造函数的构造函数对象,在这里是获取默认的构造器      final Constructor<? super T> constructor = rawType.getDeclaredConstructor();      if (!constructor.isAccessible()) {//设置访问权限        constructor.setAccessible(true);      }      return new ObjectConstructor<T>() {        @SuppressWarnings("unchecked") // T is the same raw type as is requested        public T construct() {            Object[] args = null;            //创建JavaBean的对象,并返回之            return (T) constructor.newInstance(args);          }      };  }

通过上面的代码发现该方法返回的是一个ObjectConstructor的匿名子类,通过调用该子类的construct()来创建并返回JavaBean对象。(ObjectConstructor是一个泛型接口,该接口就提供了一个construct()方法),返回的JavaBean对象交给Field的set方法第一个参数。
所以通过ReflectiveTypeAdapterFactory的create()方法返回的Adapter的read方法就是如下:

public T read(JsonReader in) throws IOException {     //此处省略了代码      //获取通过步骤2创建的JavaBean对象      T instance = constructor.construct();      try {        in.beginObject();        while (in.hasNext()) {//遍历当前JsonObject的值          //获取name,也就是json键值对的key          String name = in.nextName();          //获取这个name 绑定的JavaBean变量名          BoundField field = boundFields.get(name);          if (field == null || !field.deserialized) {            in.skipValue();          } else {            //读取此时name对应的值,设置到instance对象中            field.read(in, instance);          }        }      } catch (IllegalStateException e) {        throw new JsonSyntaxException(e);      } catch (IllegalAccessException e) {        throw new AssertionError(e);      }      in.endObject();      //返回解析完成后的JavaBean对象      return instance;    }

实际上对于Type是集合类的时候,通过反射构建集合类对象在ConstructorConstructor类的newDefaultImplementationConstructor中做了处理,这里就不多做说明了。
另外如果自定义的JavaBean中没有默认构造器,那么最终会调用ConstructorConstructor类中的newUnsafeAllocator方法来为你创建对应的JavaBean对象,如果此时创建失败的话就会抛出异常:

 throw new RuntimeException(("Unable to invoke no-args constructor for " + type + ". "              + "Register an InstanceCreator with Gson for this type may fix this problem."), e);

Gson用反射解析json ,核心简而言之就是通过反射创建JavaBean对象,循环遍历该对象的Field,并通过Field的set(JavaBean,value)方法来对JavaBean的Field赋值。其实就是简单的反射的应用,Gson只不过是做了简单而有效的封装处理来完成了这其中的操作。

1 0
原创粉丝点击