<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    West Farm
    吾本布衣,生于川北,躬耕于代碼的田地上。
    posts - 16,  comments - 15,  trackbacks - 0
    Gson is library created by google guys, it is used for java bean to json serialization/deserialization. Gson can serialize any java bean, collection, map to json, and don't need to care about the actual type of the object being processed. It also supports type level custom conversion by register custom type adapters or serializer/deserializer for a certain type. However, sometimes we just want to customize the transformation of only one filed of a java bean, but let other fields be processed as normal. consider this example below:
    public class Person {
        private String name;
        
        private int age;
     
        private boolean married;
        
        private String partner;

        ...getters and setters
    }

    If we want the "married" field to be translated to "Married" when it's value is true, otherwise "Not Married". With Gson, we can define a custom serializer for Person class to achieve this:
    public class PersonSerializer implements JsonSerializer<Person>{

        public JsonElement serialize(Person person, Type typeOfSrc,
                JsonSerializationContext context) {
            JsonObject jPerson = new JsonObject();
            jPerson.addProperty("name", person.getName());
            jPerson.addProperty("age", person.getAge());
            jPerson.addProperty("married", person.isMarried() ? "Married" : "Not Married");
            jPerson.addProperty("partner", person.getPartner());
            return jPerson;
        }

    }
    As you can see, we need to write a Serializer class for a type, even we just want to apply some simple conversion rule to a single field. Can we make this in a simpler way ? Let's dig into the source code of Gson!
    Gson(final Excluder excluder, final FieldNamingStrategy fieldNamingPolicy,
          final Map<Type, InstanceCreator<?>> instanceCreators, boolean serializeNulls,
          boolean complexMapKeySerialization, boolean generateNonExecutableGson, boolean htmlSafe,
          boolean prettyPrinting, boolean serializeSpecialFloatingPointValues,
          LongSerializationPolicy longSerializationPolicy,
          List<TypeAdapterFactory> typeAdapterFactories) {
        this.constructorConstructor = new ConstructorConstructor(instanceCreators);
        this.serializeNulls = serializeNulls;
        this.generateNonExecutableJson = generateNonExecutableGson;
        this.htmlSafe = htmlSafe;
        this.prettyPrinting = prettyPrinting;

        List<TypeAdapterFactory> factories = new ArrayList<TypeAdapterFactory>();

        // built-in type adapters that cannot be overridden
        factories.add(TypeAdapters.JSON_ELEMENT_FACTORY);
        factories.add(ObjectTypeAdapter.FACTORY);

        // the excluder must precede all adapters that handle user-defined types
        factories.add(excluder);

        // user's type adapters
        factories.addAll(typeAdapterFactories);

        // type adapters for basic platform types
        factories.add(TypeAdapters.STRING_FACTORY);
        factories.add(TypeAdapters.INTEGER_FACTORY);
        factories.add(TypeAdapters.BOOLEAN_FACTORY);
        factories.add(TypeAdapters.BYTE_FACTORY);
        factories.add(TypeAdapters.SHORT_FACTORY);
        factories.add(TypeAdapters.newFactory(long.class, Long.class, longAdapter(longSerializationPolicy)));
        factories.add(TypeAdapters.newFactory(double.class, Double.class, doubleAdapter(serializeSpecialFloatingPointValues)));
        factories.add(TypeAdapters.newFactory(float.class, Float.class, floatAdapter(serializeSpecialFloatingPointValues)));
        factories.add(TypeAdapters.NUMBER_FACTORY);
        factories.add(TypeAdapters.CHARACTER_FACTORY);
        factories.add(TypeAdapters.STRING_BUILDER_FACTORY);
        factories.add(TypeAdapters.STRING_BUFFER_FACTORY);
        factories.add(TypeAdapters.newFactory(BigDecimal.class, TypeAdapters.BIG_DECIMAL));
        factories.add(TypeAdapters.newFactory(BigInteger.class, TypeAdapters.BIG_INTEGER));
        factories.add(TypeAdapters.URL_FACTORY);
        factories.add(TypeAdapters.URI_FACTORY);
        factories.add(TypeAdapters.UUID_FACTORY);
        factories.add(TypeAdapters.LOCALE_FACTORY);
        factories.add(TypeAdapters.INET_ADDRESS_FACTORY);
        factories.add(TypeAdapters.BIT_SET_FACTORY);
        factories.add(DateTypeAdapter.FACTORY);
        factories.add(TypeAdapters.CALENDAR_FACTORY);
        factories.add(TimeTypeAdapter.FACTORY);
        factories.add(SqlDateTypeAdapter.FACTORY);
        factories.add(TypeAdapters.TIMESTAMP_FACTORY);
        factories.add(ArrayTypeAdapter.FACTORY);
        factories.add(TypeAdapters.ENUM_FACTORY);
        factories.add(TypeAdapters.CLASS_FACTORY);

        // type adapters for composite and user-defined types
        factories.add(new CollectionTypeAdapterFactory(constructorConstructor));
        factories.add(new MapTypeAdapterFactory(constructorConstructor, complexMapKeySerialization));
        factories.add(new ReflectiveTypeAdapterFactory(constructorConstructor, fieldNamingPolicy, excluder));

        this.factories = Collections.unmodifiableList(factories);
      }
    code above is the invisible constructor of Gson class, this constructor always get called, no matter how you create Gson instance. It accept some parameters that control how Gson will perform serialization/deserialization and a list of user defined type adapter factories. The most important thing here
    is the list of user defined type adapter factories, it can holds instances of TypeAdapter, JsonSerializer or JsonDeserializer. For example, the PersonSerializer in previous example will be passed in, and Gson instance will use it to serialize instances of Person class. As there are so many adapter factories, how does Gson know which one to use for a certain type? Let's take a look at the code of the method Gson#getAdapter:
    public <T> TypeAdapter<T> getAdapter(TypeToken<T> type) {
        TypeAdapter<?> cached = typeTokenCache.get(type);
        if (cached != null) {
          return (TypeAdapter<T>) cached;
        }

        Map<TypeToken<?>, FutureTypeAdapter<?>> threadCalls = calls.get();
        boolean requiresThreadLocalCleanup = false;
        if (threadCalls == null) {
          threadCalls = new HashMap<TypeToken<?>, FutureTypeAdapter<?>>();
          calls.set(threadCalls);
          requiresThreadLocalCleanup = true;
        }

        // the key and value type parameters always agree
        FutureTypeAdapter<T> ongoingCall = (FutureTypeAdapter<T>) threadCalls.get(type);
        if (ongoingCall != null) {
          return ongoingCall;
        }

        try {
          FutureTypeAdapter<T> call = new FutureTypeAdapter<T>();
          threadCalls.put(type, call);

          for (TypeAdapterFactory factory : factories) {
            TypeAdapter<T> candidate = factory.create(this, type);
            if (candidate != null) {
              call.setDelegate(candidate);
              typeTokenCache.put(type, candidate);
              return candidate;
            }
          }
          throw new IllegalArgumentException("GSON cannot handle " + type);
        } finally {
          threadCalls.remove(type);

          if (requiresThreadLocalCleanup) {
            calls.remove();
          }
        }
      }
    Gson will choose the factory which don't return null for method call on TypeAdapter#create(Gson gson, TypeToken<T> type), also notice that all factories are contained in a ArrayList which is ordered, that means the first factory added in will be checked firstly. According to the order factories added in, we can find that, strings and wrapper types of primitive types and some other types will be handled before  collections and maps, and there is
    a ReflectiveTypeAdapterFactory at last, that's something we are looking for! The adapter factory for user defined types, let's dig into it:
    /*
     * Copyright (C) 2011 Google Inc.
     *
     * Licensed under the Apache License, Version 2.0 (the "License");
     * you may not use this file except in compliance with the License.
     * You may obtain a copy of the License at
     *
     *      
    http://www.apache.org/licenses/LICENSE-2.0
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     * See the License for the specific language governing permissions and
     * limitations under the License.
     
    */

    package com.google.gson.internal.bind;

    import com.google.gson.FieldNamingStrategy;
    import com.google.gson.Gson;
    import com.google.gson.JsonSyntaxException;
    import com.google.gson.TypeAdapter;
    import com.google.gson.TypeAdapterFactory;
    import com.google.gson.annotations.SerializedName;
    import com.google.gson.extend.Convertor;
    import com.google.gson.extend.annotations.Convert;
    import com.google.gson.internal.$Gson$Types;
    import com.google.gson.internal.ConstructorConstructor;
    import com.google.gson.internal.Excluder;
    import com.google.gson.internal.ObjectConstructor;
    import com.google.gson.internal.Primitives;
    import com.google.gson.reflect.TypeToken;
    import com.google.gson.stream.JsonReader;
    import com.google.gson.stream.JsonToken;
    import com.google.gson.stream.JsonWriter;
    import java.io.IOException;
    import java.lang.reflect.Field;
    import java.lang.reflect.Type;
    import java.util.LinkedHashMap;
    import java.util.Map;

    /**
     * Type adapter that reflects over the fields and methods of a class.
     
    */
    public final class ReflectiveTypeAdapterFactory implements TypeAdapterFactory {
      private final ConstructorConstructor constructorConstructor;
      private final FieldNamingStrategy fieldNamingPolicy;
      private final Excluder excluder;

      public ReflectiveTypeAdapterFactory(ConstructorConstructor constructorConstructor,
          FieldNamingStrategy fieldNamingPolicy, Excluder excluder) {
        this.constructorConstructor = constructorConstructor;
        this.fieldNamingPolicy = fieldNamingPolicy;
        this.excluder = excluder;
      }

      public boolean excludeField(Field f, boolean serialize) {
        return !excluder.excludeClass(f.getType(), serialize) && !excluder.excludeField(f, serialize);
      }

      private String getFieldName(Field f) {
        SerializedName serializedName = f.getAnnotation(SerializedName.class);
        return serializedName == null ? fieldNamingPolicy.translateName(f) : serializedName.value();
      }

      public <T> TypeAdapter<T> create(Gson gson, final TypeToken<T> type) {
        Class<? super T> raw = type.getRawType();

        if (!Object.class.isAssignableFrom(raw)) {
          return null// it's a primitive!
        }

        ObjectConstructor<T> constructor = constructorConstructor.get(type);
        return new Adapter<T>(constructor, getBoundFields(gson, type, raw));
      }

      private ReflectiveTypeAdapterFactory.BoundField createBoundField(
          final Gson context, final Field field, final String name,
          final TypeToken<?> fieldType, boolean serialize, boolean deserialize) {
        final boolean isPrimitive = Primitives.isPrimitive(fieldType.getRawType());

        // special casing primitives here saves ~5% on Android
        return new ReflectiveTypeAdapterFactory.BoundField(name, serialize, deserialize) {
          final TypeAdapter<?> typeAdapter = context.getAdapter(fieldType);
          @SuppressWarnings({"unchecked", "rawtypes"}) // the type adapter and field type always agree
          @Override void write(JsonWriter writer, Object value)
              throws IOException, IllegalAccessException {
            Object fieldValue = field.get(value);
            TypeAdapter t = new TypeAdapterRuntimeTypeWrapper(context, this.typeAdapter, fieldType.getType());
            t.write(writer, fieldValue);
          }
          @Override void read(JsonReader reader, Object value)
              throws IOException, IllegalAccessException {
            Object fieldValue = typeAdapter.read(reader);
            if (fieldValue != null || !isPrimitive) {
              field.set(value, fieldValue);
            }
          }
        };
      }

      private Map<String, BoundField> getBoundFields(Gson context, TypeToken<?> type, Class<?> raw) {
        Map<String, BoundField> result = new LinkedHashMap<String, BoundField>();
        if (raw.isInterface()) {
          return result;
        }

        Type declaredType = type.getType();
        while (raw != Object.class) {
          Field[] fields = raw.getDeclaredFields();
          for (Field field : fields) {
            boolean serialize = excludeField(field, true);
            boolean deserialize = excludeField(field, false);
            if (!serialize && !deserialize) {
              continue;
            }
            field.setAccessible(true);
            Type fieldType = $Gson$Types.resolve(type.getType(), raw, field.getGenericType());
            BoundField boundField = createBoundField(context, field, getFieldName(field),
                TypeToken.get(fieldType), serialize, deserialize);
            BoundField previous = result.put(boundField.name, boundField);
            if (previous != null) {
              throw new IllegalArgumentException(declaredType
                  + " declares multiple JSON fields named " + previous.name);
            }
          }
          type = TypeToken.get($Gson$Types.resolve(type.getType(), raw, raw.getGenericSuperclass()));
          raw = type.getRawType();
        }
        return result;
      }

      static abstract class BoundField {
        final String name;
        final boolean serialized;
        final boolean deserialized;

        protected BoundField(String name, boolean serialized, boolean deserialized) {
          this.name = name;
          this.serialized = serialized;
          this.deserialized = deserialized;
        }

        abstract void write(JsonWriter writer, Object value) throws IOException, IllegalAccessException;
        abstract void read(JsonReader reader, Object value) throws IOException, IllegalAccessException;
      }

      public static final class Adapter<T> extends TypeAdapter<T> {
        private final ObjectConstructor<T> constructor;
        private final Map<String, BoundField> boundFields;

        private Adapter(ObjectConstructor<T> constructor, Map<String, BoundField> boundFields) {
          this.constructor = constructor;
          this.boundFields = boundFields;
        }

        @Override public T read(JsonReader in) throws IOException {
          if (in.peek() == JsonToken.NULL) {
            in.nextNull();
            return null;
          }

          T instance = constructor.construct();

          try {
            in.beginObject();
            while (in.hasNext()) {
              String name = in.nextName();
              BoundField field = boundFields.get(name);
              if (field == null || !field.deserialized) {
                in.skipValue();
              } else {
                field.read(in, instance);
              }
            }
          } catch (IllegalStateException e) {
            throw new JsonSyntaxException(e);
          } catch (IllegalAccessException e) {
            throw new AssertionError(e);
          }
          in.endObject();
          return instance;
        }

        @Override public void write(JsonWriter out, T value) throws IOException {
          if (value == null) {
            out.nullValue();
            return;
          }

          out.beginObject();
          try {
            for (BoundField boundField : boundFields.values()) {
              if (boundField.serialized) {
                out.name(boundField.name);
                boundField.write(out, value);
              }
            }
          } catch (IllegalAccessException e) {
            throw new AssertionError();
          }
          out.endObject();
        }
      }
    }
    It's just as you ever imagined, Gson use java reflection API to serialize/deserialize java beans. Please notice the method ReflectiveTypeAdapterFactory.BoundField#write(JsonWriter writer, Object value), this is where a field of a java bean being translated into json, and we can extend Gson library from here, let's add some code into it:
    void write(JsonWriter writer, Object value) throws IOException,
                        IllegalAccessException, GsonExtensionException {
                    Object fieldValue = field.get(value);
                    TypeAdapter t = new TypeAdapterRuntimeTypeWrapper(context, this.typeAdapter, fieldType.getType());
                    
                    if (field.isAnnotationPresent(Convert.class)) {
                        Convert anno = field.getAnnotation(Convert.class);

                        String exp = anno.toJsonExpression();
                        if(exp != null && exp.length() > 0){
                            Map<String, Object> vars = new HashMap<String, Object>();
                            vars.put("object", value);
                            vars.put("field", fieldValue);
                            fieldValue = MVEL.eval(exp, vars);
                        }else{
                            String convertorClass = anno.toJsonConvertor();
                            if(!convertorClass.equals("")){
                                try {
                                    Object obj = Class.forName(convertorClass).newInstance();
                                    if(obj instanceof Convertor){
                                        Convertor convertor = (Convertor)obj;
                                        fieldValue = convertor.Convert(fieldValue);
                                    }else{
                                        throw new GsonExtensionException("the convertor class [" + convertorClass + "] does't implement " + "the cn.ggfan.gson.ex.Convertor interface.");
                                    }
                                } catch (InstantiationException e) {
                                    e.printStackTrace();
                                    throw new GsonExtensionException("the convertor class [" + convertorClass + "] can't be instantiated: " + e.getMessage());
                                } catch (ClassNotFoundException e) {
                                    e.printStackTrace();
                                    throw new GsonExtensionException("the convertor class [" + convertorClass + "] can't be found on classpath: " + e.getMessage());
                                }
                            }
                        }
                    }
                    
                    t.write(writer, fieldValue);
                }
    the annotation:
    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.FIELD)
    public @interface Convert {
        
        public String toJsonConvertor() default "";
        
        public String toJsonExpression() default "";
        
        public String fromJsonConvertor() default "";
        
        public String fromJsonExpression() default "";

    }
    now annotated Person class:
    package cn.ggfan.gson.ex.test.model;

    import cn.ggfan.gson.ex.annotations.Convert;

    public class Person {
        private String name;
        
        private int age;
        
        @Convert(
            toJsonExpression = "field ? '已婚' : '未婚'", 
            fromJsonExpression = "field.equals(\"已婚\") ? true : false"
        )
        private boolean married;
        
        private String partner;

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public int getAge() {
            return age;
        }

        public void setAge(int age) {
            this.age = age;
        }

        public boolean isMarried() {
            return married;
        }

        public void setMarried(boolean married) {
            this.married = married;
        }

        public String getPartner() {
            return partner;
        }

        public void setPartner(String partner) {
            this.partner = partner;
        }

        @Override
        public String toString() {
            return "Person [name=" + name + ", age=" + age + ", married=" + married
                    + ", partner=" + partner + "]";
        }

    }
    the expression used in our annotation is MVEL expressions, I will not explain it here. It just interpret some expression to a value. 
    Finally, let's see some example:
            Person girl = new Person();
            girl.setName("Li Ying");
            girl.setAge(24);
            girl.setMarried(false);
            girl.setPartner("Liu Dehua");
            System.out.println(new Gson().toJson(girl));
    output :
    {
      "name": "Li Ying",
      "age": 24,
      "married": "未婚",
      "partner": "Liu Dehua"
    }
    don't be confused, the Chinese word "未婚" means "Not Married", and "已婚" means "Married" :D

    posted on 2013-11-22 19:22 West Farmer 閱讀(5333) 評(píng)論(3)  編輯  收藏 所屬分類: Java

    FeedBack:
    # re: Extend Gson to suppurt field level custom conversion
    2013-11-22 19:29 | West Farmer
    you can even convert a primitive field to another java bean, for example, the partner field can be replaced with a Person instance!  回復(fù)  更多評(píng)論
      
    # re: Extend Gson to suppurt field level custom conversion
    2013-12-03 11:24 | West Farmer
    And I find it's more useful than intended :D, for example we can add support for the javax.persistence.Transient annotation, that means ignore any fields annotated with Transient.  回復(fù)  更多評(píng)論
      
    # re: Extend Gson to support field level custom conversion
    2014-09-02 05:45 | Martyn
    This is interesting - do you have a working example you can share?

    Thanks  回復(fù)  更多評(píng)論
      

    <2013年11月>
    272829303112
    3456789
    10111213141516
    17181920212223
    24252627282930
    1234567

    常用鏈接

    留言簿

    隨筆分類

    隨筆檔案

    相冊

    搜索

    •  

    最新評(píng)論

    閱讀排行榜

    評(píng)論排行榜

    主站蜘蛛池模板: 亚洲午夜精品一区二区公牛电影院| 男女午夜24式免费视频| 亚洲精品无码你懂的网站| 免费毛片在线看不用播放器 | 亚洲视频免费播放| 在线观看国产情趣免费视频| 在线观看免费无码视频| 亚洲欧美日韩综合久久久| 亚洲av无码一区二区三区乱子伦| 午夜色a大片在线观看免费| 久久狠狠躁免费观看| 国产在线播放线91免费| 欧美亚洲精品一区二区| 亚洲国产最大av| 亚洲国产成AV人天堂无码| 久久久久亚洲AV成人片| 亚洲国产一成人久久精品| 中文字幕无码精品亚洲资源网| 成人免费a级毛片无码网站入口 | 国产亚洲精品免费视频播放| 日本免费电影一区| 国产高清免费观看| 成年女人18级毛片毛片免费| 69式国产真人免费视频| 男女超爽刺激视频免费播放| 在线天堂免费观看.WWW| 日本午夜免费福利视频| 老司机永久免费网站在线观看| 国产三级在线观看免费| 国产免费黄色大片| 亚洲午夜福利在线观看| 久久亚洲中文字幕精品有坂深雪 | 国产中文字幕在线免费观看| 国产精品无码免费专区午夜| 国产高清对白在线观看免费91 | 亚洲国产日韩成人综合天堂| 亚洲中文字幕在线乱码| 亚洲理论片在线观看| 免费手机在线看片| 久久午夜免费视频| 国产偷窥女洗浴在线观看亚洲|