TypeReference-FastJson详解

概述

TypeReference是一个描述复杂泛型的工具类。
很多类库都有,我们使用alibaba.fastjson来举例。

作用

在反序列化场景中获取泛型参数。
TypeReference支持泛型参数,方便一些框架实现通用的反序列化类,对复杂的类型可以很方便的反序列化。
使用Gson、Jackson或Fastjson反序列化泛型时,需要传递泛型的真实类型,所以一般都通过集成TypeReference来实现。

示例

简单示例

@Slf4j(topic = "juc")
public class TypeRefrenceStudy {
    public static void main(String[] args) {
        List<Integer> list = Lists.newArrayList(1, 3, 4, 8, 9);
        JSONObject jsonObject = new JSONObject();
        jsonObject.put("list", list);
        log.info(jsonObject.toString());

		// 使用匿名类创建TypeReference的子类对象
        List<Integer> list2 = jsonObject.getObject("list", new TypeReference<List<Integer>>() {});
        log.info("list2 {}", list2);
    }
}

执行结果:

16:07:37.122 [main] juc - {"list":[1,3,4,8,9]}
16:07:37.141 [main] juc - list2 [1, 3, 4, 8, 9]
  • 使用TypeReference可以明确指定反序列化的类型,能够比较方便的获取反序列化的数据。

复杂示例

例子:Response response = JSONObject.parseObject(result, new TypeReference<Response>(TaskCodeRespData.class) {});
等价于:
Response response = JSONObject.parseObject(result, new TypeReference<Response>( ) {})
如果要用泛型变量 ,有参数的写法就相当必要了。

单参数例子

public class Response {
public T data;
}
public static Response parseToMap(String json, Class type) {
return JSON.parseObject(json,
new TypeReference<Response>(type) {});
}

双参数例子

public static <K, V> Map<K, V> parseToMap(String json,
Class keyType,
Class valueType) {
return JSON.parseObject(json,
new TypeReference<Map<K, V>>(keyType, valueType) {
});
}

// 可以这样使用
String json = “{1:{name:“ddd”},2:{name:“zzz”}}”;
Map<Integer, Model> map = parseToMap(json, Integer.class, Model.class);
assertEquals(“ddd”, map.get(1).name);
assertEquals(“zzz”, map.get(2).name);

源码解析

无参构造器

    protected TypeReference() {
        Type superClass = this.getClass().getGenericSuperclass();
        Type type = ((ParameterizedType)superClass).getActualTypeArguments()[0];
        Type cachedType = (Type)classTypeCache.get(type);
        if (cachedType == null) {
            classTypeCache.putIfAbsent(type, type);
            cachedType = (Type)classTypeCache.get(type);
        }

        this.type = cachedType;
    }
  • 可以看到构造函数是protect的,所以只能使用TypeReference的继承类生成对象。

ParameterizedType解析

ParameterizedType用来表示一个泛型类型(泛型类的模板中有至少一个类型参数),比如Collection。ParameterizedType继承Type,共有3个方法。

  • Type[] getActualTypeArguments();// 返回类的泛型类型数组。
  • Type getRawType();// 返回原始类型Type,类型本身的类型。
  • Type getOwnerType();// 返回Type对象,表示此类型是其成员之一的类型。

例如 Map<String,String> 对应的ParameterizedType三个方法分别取值如下:
[class java.lang.String, class java.lang.String]
interface java.util.Map
null

测试

package JsonLearn;
 
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.HashMap;
import java.util.Map;
 
public class TypeReferencBaseLearn {
    public static class IntMap extends HashMap<String, Integer> {}
 
    void test1() {
        IntMap intMap = new IntMap();
        System.out.println("getSuperclass:" + intMap.getClass().getSuperclass());
        System.out.println("getGenericSuperclass:" + intMap.getClass().getGenericSuperclass());
        Type type = intMap.getClass().getGenericSuperclass();
        if (type instanceof ParameterizedType) {
            ParameterizedType p = (ParameterizedType)type;
            for (Type t : p.getActualTypeArguments()) {
                System.out.println(t);
            }
        }
    }
 
    void test2() {
        Map<String, Integer> intMap = new HashMap<>();
        System.out.println("\ngetSuperclass:" + intMap.getClass().getSuperclass());
        System.out.println("getGenericSuperclass:" + intMap.getClass().getGenericSuperclass());
        Type type = intMap.getClass().getGenericSuperclass();
        if (type instanceof ParameterizedType) {
            ParameterizedType p = (ParameterizedType)type;
            for (Type t : p.getActualTypeArguments()) {
                System.out.println(t);
            }
        }
    }
 
    void test3() {
        Map<String, Integer> intMap = new HashMap<String, Integer>(){};
        System.out.println("\ngetSuperclass:" + intMap.getClass().getSuperclass());
        System.out.println("getGenericSuperclass:" + intMap.getClass().getGenericSuperclass());
        Type type = intMap.getClass().getGenericSuperclass();
        if (type instanceof ParameterizedType) {
            ParameterizedType p = (ParameterizedType)type;
            for (Type t : p.getActualTypeArguments()) {
                System.out.println(t);
            }
        }
    }
 
    public static void main(String[] args) {
        TypeReferencBaseLearn obj = new TypeReferencBaseLearn();
        obj.test1();
        obj.test2();
        obj.test3();
    }
}

输出

// test1
getSuperclass:class java.util.HashMap
getGenericSuperclass:java.util.HashMap<java.lang.String, java.lang.Integer>
class java.lang.String
class java.lang.Integer

// test2 
getSuperclass:class java.util.AbstractMap
getGenericSuperclass:java.util.AbstractMap<K, V>
K
V
 
// test2
getSuperclass:class java.util.HashMap
getGenericSuperclass:java.util.HashMap<java.lang.String, java.lang.Integer>
class java.lang.String
class java.lang.Integer 

参考

Java中TypeReference用法说明

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
Fastjson是一个Java语言编写的高性能JSON处理框架,它提供了简单易用的API,可以实现Java对象和JSON字符串之间的相互转换。下面是一些关于Fastjson详解: 1. 特性:Fastjson具有较高的性能和较低的内存占用,支持全序列化和全反序列化,支持复杂对象、集合和泛型的序列化与反序列化,支持自定义序列化和反序列化处理逻辑,支持处理循环引用等。 2. 使用方法:Fastjson提供了简单易用的API,可以通过对象的toJsonString()方法将对象转换为JSON字符串,通过parseObject()方法将JSON字符串转换为对象。可以使用注解 @JSONField 进行字段级别的配置,还可以通过设置全局配置类 JSONConfig 来对Fastjson进行全局配置。 3. 序列化与反序列化:Fastjson可以将Java对象序列化为JSON字符串,也可以将JSON字符串反序列化为Java对象。在序列化过程中,Fastjson会自动根据对象的类型进行转换,并且支持复杂对象、集合和泛型等数据结构。在反序列化过程中,Fastjson会根据JSON字符串的结构自动将其转换为对应的Java对象。 4. 高级特性:Fastjson提供了一些高级特性,如支持处理循环引用,可以通过设置SerializerFeature.DisableCircularReferenceDetect来禁用循环引用检测;支持处理日期格式化,可以通过设置DateFormat来指定日期格式;支持处理泛型类型,可以通过TypeReference来指定泛型类型。 总的来说,Fastjson是一个功能强大、性能高效的JSON处理框架,在Java开发中被广泛使用。它简化了Java对象和JSON字符串之间的转换过程,并提供了丰富的功能和灵活的配置选项。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

融极

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值