com.google.gson.internal.LinkedTreeMap cannot be cast to XXX

文章讲述了作者在本地缓存Google商品时遇到的bug,通过逐步排查发现是由于GoogleSDK中的List没有添加泛型导致Gson反序列化失败。最后采取了类型判断和自定义序列化/反序列化代码的解决方案。

起因是在对google商品做本地缓存时,上线后发现的bug

刚开始非常自信,debug没问题线上有问题,大概率就是混淆文件没有添加keep,于是本地添加对SDK中类的keep,本地打包release验证,不出意外还是崩溃

仔细看崩溃的原因,关键字是com.google.gson.internal.LinkedTreeMap ,仔细想想就是gson做反序列化时无法转化为我们的类型,于是检查gson的语法,以为是两层List<List< T >>导致,去google查询按照网上的代码修改序列化和反序列化的代码还是不行。

在这里插入图片描述
最终定位原因是Google 提供的SDK中声明的List没有添加泛型!导致Gson反序列化的时候不知道转化成什么类型
在这里插入图片描述
在这里插入图片描述

因为没办法修改SDK,最终用了一种tricky的解决方案,在获取到对象的时候做个类型判断,如果是com.google.gson.internal.LinkedTreeMap类型的话就用map去取对应的字段,即map.get("XX")

另外写一个通用的Gson序列化和反序列化的代码

// 序列化   把ProductDetails换成对应的类型
kotlin.runCatching {
    Gson().toJson(it, object : TypeToken<List<ProductDetails>>() {}.type)
}
// 反序列化  把ProductDetails换成对应的类型   priceJson换成你要反序列化的对象
try {
    Gson().fromJson<List<ProductDetails>>(priceJson, object : TypeToken<List<ProductDetails>>() {}.type)
} catch (e: Exception) {
    e.throwWhenLog()
}
### 解决 `LinkedTreeMap` 无法转换为 `String` 当使用 Gson 进行 JSON 反序列化时,如果目标对象是一个自定义类而非标准集合类型,则可能会遇到 `ClassCastException`。具体来说,在尝试将 `com.google.gson.internal.LinkedTreeMap` 转换为其他类型(如 `java.lang.String` 或者任何特定业务实体类)时会抛出此异常。 为了防止这种错误发生,可以采取以下几种方法之一: #### 方法一:确保正确指定泛型参数 在调用 `fromJson()` 函数之前,应该通过 `TypeToken` 明确指出期望的结果类型。这有助于让 Gson 知道如何构建最终的对象实例而不是默认创建 `LinkedTreeMap` 实例来表示未知类型的映射关系[^3]。 ```java // 定义具体的返回类型 Type typeOfT = new TypeToken<String>(){}.getType(); String resultStr = gson.fromJson(jsonInput, typeOfT); ``` #### 方法二:处理原始 Map 数据再转为目标类型 另一种方式是从 JSON 中先解析得到一个通用的 `Map<String,Object>` 对象,然后再根据实际情况将其进一步加工成为所需的字符串或其他形式的数据结构[^1]。 ```java JsonElement jsonElement = JsonParser.parseString(jsonInput); if (jsonElement.isJsonObject()) { JsonObject jsonObject = jsonElement.getAsJsonObject(); Set<Map.Entry<String, JsonElement>> entries = jsonObject.entrySet(); StringBuilder sb = new StringBuilder(); for (Map.Entry<String, JsonElement> entry : entries) { sb.append(entry.getKey()).append(": ").append(entry.getValue().getAsString()).append("\n"); } String formattedResult = sb.toString(); } ``` #### 方法三:编写自定义反序列化器 对于更复杂的需求,还可以考虑实现自己的 `JsonDeserializer<T>` 接口来自定义反序列化的逻辑流程,从而更好地控制从 JSON 到 Java 对象之间的转换过程[^2]。 ```java public class CustomDeserializer implements JsonDeserializer<String> { @Override public String deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException { // 自定义转换逻辑... return json.getAsString(); } } GsonBuilder builder = new GsonBuilder(); builder.registerTypeAdapter(String.class, new CustomDeserializer()); Gson customGson = builder.create(); customGson.fromJson(jsonInput, String.class); ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值