Gson解析 com.google.gson.internal.LinkedTreeMap cannot be cast to XXX

本文讲述了如何修复在将ArrayList通过Gson保存到SharedPreferences后,从SharedPreferences读取时遇到的类型转换问题。作者分享了修正方法,即正确使用`newTypeToken`和指定目标类型来确保解析正确。

问题描述:项目从网络获取数据保存到sharedpreferences,再从sharedpreferences获取。由于要保存到shp的是arraylist,所以需要使用Gson解析,在解析过程中报了如下错误:
com.google.gson.internal.LinkedTreeMap cannot be cast to XXX

原先代码如下:
将list转化为json字符串保存在本地:

  /**
     * 保存List
     */
    public static <T> void setDataList(String tag, List<T> datalist, Context context) {
        if (null == datalist || datalist.size() <= 0)
            return;

        Gson gson = new Gson();
        //转换成json数据,再保存
        String strJson = gson.toJson(datalist);
        SharedPreferences.Editor editor = getKeywordPreference(context).edit();
        editor.clear();
        editor.putString(tag, strJson);
        editor.apply();

    }

将保存在shp的字符串取出,重新解析为ArrayList类型:

    /**
     * 获取List
     */
    public static <T> List<T> getDataList(Class<T> clazz, String tag, Context context) {
        List<T> datalist = new ArrayList<>();
        String strJson = getKeywordPreference(context).getString(tag, null);
        if (null == strJson) {
            return datalist;
        }
        Gson gson = new Gson();
        datalist = gson.fromJson(strJson, new TypeToken<List<T>>() {}.getType());
        return datalist;
    }

运行后报错,根据报错信息可知,代码并没有成功把json字符串解析为我们想要的类型,而是解析为了LinkedTreeMap,所以问题就出在获取list的方法中的这一句:

datalist = gson.fromJson(strJson,new TypeToken<List<T>>(){}.getType());

由此可知

解决方法

只需要将获取type的方法修改一下即可:

    /**
     * 获取List
     */
    public static <T> List<T> getDataList(Class<T> clazz, String tag, Context context) {
        List<T> datalist = new ArrayList<>();
        String strJson = getKeywordPreference(context).getString(tag, null);
        if (null == strJson) {
            return datalist;
        }
        Gson gson = new Gson();
        //        datalist = gson.fromJson(strJson, new TypeToken<List<T>>() {}.getType());
        //改为下面的方法,clazz传入实际想要解析出来的类
        Type listType = com.google.gson.internal.$Gson$Types.newParameterizedTypeWithOwner(null, ArrayList.class, clazz);
        datalist = gson.fromJson(strJson, listType);
        return datalist;
    }
### 解决 `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); ```
评论 7
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值