- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
如何让 Gson 反序列化计算值?
这是一个过于简单化的示例:
@SerializedName("members")
@Expose
private final List<String> members;
@SerializedName("size")
@Expose(serialize = true, deserialize = false)
private final int size;
public Club(List<String> members) {
this.members = members;
this.size = members.size();
}
当deserialize
为false
时,我得到size
= 0;当 deserialize
为 true
时,我得到 size
= old_size。
有没有办法“强制”重新计算?
我知道我可以在 getter 中执行此操作
最佳答案
发生这种情况是因为 Gson 在反序列化期间不会调用构造函数。但是,可以在实例化后阶段拦截对象,并对最近反序列化的对象进行一些附加操作。
这是一个示例@PostConstruct
感知类型适配器工厂:
final class PostConstructTypeAdapterFactory
implements TypeAdapterFactory {
// No intermediate state, can be a singleton
private static final TypeAdapterFactory postConstructTypeAdapterFactory = new PostConstructTypeAdapterFactory();
private PostConstructTypeAdapterFactory() {
}
// However, making the constructor private encapsulates the way it's instantiated
static TypeAdapterFactory getPostConstructTypeAdapterFactory() {
return postConstructTypeAdapterFactory;
}
@Override
public <T> TypeAdapter<T> create(final Gson gson, final TypeToken<T> typeToken) {
final List<Method> postConstructMethods = getPostConstructMethods(typeToken.getRawType());
if ( postConstructMethods.isEmpty() ) {
// If no post-construct methods found, just let Gson to pick up the next best-match type adapter itself
return null;
}
// Obtain the "original" type adapter
final TypeAdapter<T> delegateTypeAdapter = gson.getDelegateAdapter(this, typeToken);
return new PostConstructTypeAdapter<>(delegateTypeAdapter, postConstructMethods);
}
private static List<Method> getPostConstructMethods(final Class<?> clazz) {
if ( clazz.isPrimitive() ) {
// Nothing to do with primitives
return emptyList();
}
// The following stream operation collects all `@PostConstruct` methods
// from java.lang.Object to the concrete class
// where all @PostConstruct methods must satisfy the following conditions (differ from the original `@PostConstruct` contract):
// * have no paramaters
// * return nothing (but it looks like a too strict limitation, though)
// * be annotated with `@PostConstruct`
return superToSub(clazz)
.stream()
.flatMap(c -> Stream.of(c.getDeclaredMethods()))
.filter(m -> {
final int parameterCount = m.getParameterCount();
if ( parameterCount != 0 ) {
return false;
}
final Class<?> returnType = m.getReturnType();
if ( returnType != void.class ) {
return false;
}
return m.isAnnotationPresent(PostConstruct.class);
})
.peek(m -> m.setAccessible(true))
.collect(toList());
}
private static List<Class<?>> superToSub(final Class<?> clazz) {
final List<Class<?>> hierarchy = subToSuper(clazz);
Collections.reverse(hierarchy);
return hierarchy;
}
private static List<Class<?>> subToSuper(final Class<?> clazz) {
final List<Class<?>> hierarchy = new ArrayList<>();
for ( Class<?> c = clazz; c != null; c = c.getSuperclass() ) {
hierarchy.add(c);
}
return hierarchy;
}
private static final class PostConstructTypeAdapter<T>
extends TypeAdapter<T> {
private final TypeAdapter<T> delegateTypeAdapter;
private final Iterable<Method> postConstructMethods;
private PostConstructTypeAdapter(final TypeAdapter<T> delegateTypeAdapter, final Iterable<Method> postConstructMethods) {
this.delegateTypeAdapter = delegateTypeAdapter;
this.postConstructMethods = postConstructMethods;
}
@Override
public void write(final JsonWriter out, final T value)
throws IOException {
// Nothing special to do on write, so just delegate the job
delegateTypeAdapter.write(out, value);
}
@Override
public T read(final JsonReader in)
throws IOException {
try {
// Whilst on read there's a need to apply all post-construction methods
final T read = delegateTypeAdapter.read(in);
for ( final Method method : postConstructMethods ) {
method.invoke(read);
}
return read;
} catch ( IllegalAccessException | InvocationTargetException ex ) {
throw new IOException(ex);
}
}
}
}
然后,您的 Club
可以包含后构造函数:
@PostConstruct
private void postConstruct()
throws NoSuchFieldException, IllegalAccessException {
final Field sizeField = Club.class.getDeclaredField("size");
sizeField.setAccessible(true);
sizeField.set(this, members.size());
}
临时测试的使用示例:
private static final Gson gson = new GsonBuilder()
.excludeFieldsWithoutExposeAnnotation()
.registerTypeAdapterFactory(getPostConstructTypeAdapterFactory())
.create();
public static void main(final String... args) {
final List<String> members = ImmutableList.of("Foo", "Bar", "Baz");
final Club beforeClub = new Club(members);
final List<String> beforeMembers = beforeClub.members;
final int beforeSize = beforeClub.size;
final String clubJson = gson.toJson(beforeClub);
System.out.println(clubJson);
final Club afterClub = gson.fromJson(clubJson, Club.class);
final List<String> afterMembers = afterClub.members;
final int afterSize = afterClub.size;
if ( !beforeMembers.equals(afterMembers) ) {
throw new AssertionError("`members` values do not match");
}
if ( beforeSize != afterSize ) {
throw new AssertionError("`size` values do not match");
}
System.out.println("SUCCESS");
}
输出:
{"members":["Foo","Bar","Baz"],"size":3}
SUCCESS
查看更多:
@PostConstruct
support在官方 Gson 存储库中。@PostConstruct
Javadoc .关于java - Gson:反序列化一个计算值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43359657/
我试图让 gson 在 java 项目上工作,但每次运行时都会出现上述错误。我没有使用任何闪存 IDE,只是 vim,我看到的与我的问题相关的每个问题都与 eclipse 及其部署设置有关。我希望有人
我一直在关注将对象数组转换为 JSON 的教程,但我遇到了一些找不到解决方案的错误。 代码 第 60 - 64 行 Gson gson = new Gson().toJson(data); respo
我正在创建一个新的 Netty 管道,我正在尝试: 避免过早优化。 编写易于向我的一位实习生解释的代码。 这种工厂方法当然很容易解释: public String toJSON() { Gso
我尝试在程序中使用 Unirest,但不断收到此错误 java.lang.NoSuchMethodError: com.google.gson.Gson.newBuilder()Lcom/google
// Retrofit compile 'com.squareup.retrofit2:retrofit:2.1.0' // JSON Parsing compile 'com.google.code
我正在尝试将 Gson 与接口(interface)一起使用: public interface Photo { public int getWidth(); } public class D
我正在执行一个 Web 服务,这是我的响应,但是当我尝试使用 Gson 库将此 JSONObject(org.json.JSONObject) 转换为特定对象时,我的应用程序崩溃了。所以,我不知道为什
我有以下 .json 文件 - { "IDs":[ "1716136233", "2030187302", "204897807
我试图在我的项目中使用 GSON,但我的应用程序崩溃了,logcat 说找不到 com.google.gson.Gson。我已将 import com.google.gson.Gson 放在我的类文件
在下面的示例中,我尝试序列化我的自定义类Couple,其中包含Point2D 类型的字段。 我发现,SErialzing 的过程取决于 DEserializer 的存在。 此外,如果存在反序列化器,则
我有课 class ThreadComment( banned: Int, closed: Int, comment: String?, date: String?,
使用 gson 解析 json 时,我有一个奇怪的行为。我使用这个代码: private static Container parseContainer(String containerJson) {
我注意到 GSON HTML 转义 字符,可以通过使用disableHtmlEscaping()来禁用它构建器配置方法。但我的问题是 - 为什么 GSON 默认情况下会进行 HTML 转义?不进行 H
我注意到一个奇怪的问题。我可以使用 Junit 运行我的测试用例,但是当我使用 maven 运行时,其中一个测试用例失败。它提示找不到 Gson 类 def。 我能够在 Maven 依赖项中看到 Gs
我有一个看起来像这样的 json {response:{"status":{"....."},data:[{"name":"Alice","id":"123"},{"name":"Jack","id"
所以,我的应用已经发布了将近一年,但没有看到这个问题,现在它出现了。 即使是现在,我的手机上的调试版本也没有这个问题。我对从 Android Studio 打开的任何模拟器没有任何问题。然而,Goog
我是 Android 开发环境的新手,因此需要专家的帮助。 java.lang.NoClassDefFoundError: com.google.gson.Gson 项目中包含 Gson 库,但是当从
总结 我在我的 android 应用程序中使用 Retrofit 和 Gson 有一段时间了,但是我从服务器获得的 ID 是 13 位数字。 Gson 自动将这些数字转换为科学计数法,然后将其转换为
我正在使用一个大的 JSON 对象,它具有来自多个请求的响应。 我正在处理的部分只需要很少的对象,而且它们并不总是在前面。例如 json 结构是: ** json = { mainDocume
我有一个休息网络服务: @Path("/tranreq") public class TranscriptRequesterResource { private static final Gs
我是一名优秀的程序员,十分优秀!