我在持久对象上使用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> {
}
基本上我无法同时使用这两种功能。任何可能导致此问题的指针?
答案 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人报告,他们将在下一个版本中解决它。