Gson排除策略和自定义序列化

时间:2012-11-10 01:59:08

标签: java gson

我在持久对象上使用Gson自定义序列化。同时,我也在使用序列化排除策略。代码如下所示:

public class GsonFactory {

    public static Gson build(Type type, final List<String> fieldExclusions,
            final List<Class<?>> classExclusions) {
        GsonBuilder gsonBuilder = new GsonBuilder();
        gsonBuilder.addSerializationExclusionStrategy(new ExclusionStrategy() {
            public boolean shouldSkipField(FieldAttributes f) {
                return fieldExclusions == null ? false : fieldExclusions
                        .contains(f.getName());
            }

            public boolean shouldSkipClass(Class<?> clazz) {
                return classExclusions == null ? false : classExclusions
                        .contains(clazz);
            }
        });
        // Uncommenting this line will produce error
        // gsonBuilder.registerTypeAdapter(type,
        // new PersistentObjectJsonSerializer());
        return gsonBuilder.create();
    }
}

class PersistentObjectJsonSerializer implements
        JsonSerializer<PersistentObject> {

    public JsonElement serialize(PersistentObject src, Type typeOfSrc,
            JsonSerializationContext context) {
        src.setDT_RowId(src.getId());
        Gson gson = new Gson();
        return gson.toJsonTree(src);
    }

}

但是,如果我取消注释gsonBuilder.registerTypeAdapter(type, new PersistentObjectJsonSerializer());,则在创建gsonBuilder时会出现以下错误:

java.lang.StackOverflowError
    com.google.gson.reflect.TypeToken.equals(TypeToken.java:284)
    java.util.HashMap.get(HashMap.java:305)
    java.util.Collections$SynchronizedMap.get(Collections.java:1979)
    com.google.gson.Gson.getAdapter(Gson.java:337)
    com.google.gson.internal.bind.TypeAdapterRuntimeTypeWrapper.write(TypeAdapterRuntimeTypeWrapper.java:55)
    com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$1.write(ReflectiveTypeAdapterFactory.java:89)
    com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.write(ReflectiveTypeAdapterFactory.java:195)
    com.google.gson.Gson$FutureTypeAdapter.write(Gson.java:883)

My PersistentObject java class:

@MappedSuperclass
public abstract class PersistentObject implements Serializable {

....

@Transient
protected long DT_RowId;
// getters, setters not shown

...

}

这就是我在GenericHibernateDAO中调用GsonFactory.build的方式:

public abstract class GenericHibernateDAO<T, ID extends Serializable> {

private Class<T> persistentClass;

@SuppressWarnings("unchecked")
public GenericHibernateDAO() {
    this.persistentClass = (Class<T>) ((ParameterizedType) getClass()
            .getGenericSuperclass()).getActualTypeArguments()[0];
}


public String listAsJsonResponse(){
    ...
    // testing start
    List<Class<?>> classExclusions = new ArrayList<Class<?>>();
    classExclusions.add(Role.class);
    List<String> fieldExclusions = new ArrayList<String>();
    fieldExclusions.add("users");
    fieldExclusions.add("password");
    Gson gson = GsonFactory.build(persistentClass,fieldExclusions, null);
    ...
    return jsonResponse.toString();
}
}

这里的persistentClass是指User类:

@Repository
public class UserDAO extends GenericHibernateDAO<User, Long> {

}

基本上我无法同时使用这两种功能。任何可能导致此问题的指针?

2 个答案:

答案 0 :(得分:2)

这是一个老问题,但也许我的答案仍然可以帮助某人。

您在序列化程序中创建一个新的Gson对象。此Gson对象不再知道PersistentObjects的ExclusionStrategies。如果您的PersistantObject具有将再次导致PersistentObject的属性对象(在您的情况下,用户可能具有persistantObject的属性),那么您将遇到无限循环并进入stackoverflow异常。

一种解决方案是再次将排除策略添加到序列化程序中的gson对象。但是不要再将序列化器添加到这个gson对象中了!

class PersistentObjectJsonSerializer implements
    JsonSerializer<PersistentObject> {

    public JsonElement serialize(PersistentObject src, Type typeOfSrc,
        JsonSerializationContext context) {
        src.setDT_RowId(src.getId());
        Gson gson = new Gson();
        //add exclusion strategy here again
        return gson.toJsonTree(src);
    }

}

答案 1 :(得分:0)

Gson是开源的,你可以获得源代码。为什么不这样做并按照代码查看它抛出异常的哪一点?似乎gson陷入无限循环。确实,gson经常使用递归。如果它实际上是一个错误,你可以向Gson人报告,他们将在下一个版本中解决它。

http://code.google.com/p/google-gson/issues/list