TypeReference 泛型的使用场景及具体使用流程

简介

在 Java 中,泛型类型在运行时会被擦除。这意味着当我们使用泛型时,运行时无法直接获取到泛型的具体类型信息 。例如,我们无法直接通过 Class 对象来获取一个泛型类型的类型参数。这在某些情况下可能会导致问题,特别是在我们需要对泛型类型进行反射操作时。

TypeReference 是 Jackson 库提供的一个类,它可以帮助我们解决这个问题。通过 TypeReference,我们可以在运行时保留泛型的类型信息,从而能够正确地处理复杂的泛型类型。

本文将详细介绍 TypeReference 泛型的使用场景及具体使用流程,并通过一个实际例子展示如何在自定义 HttpClient 中使用 TypeReference

使用场景

处理复杂泛型的 JSON 反序列化

在使用 Jackson 库进行 JSON 反序列化时,如果目标类型是一个复杂的泛型类型,直接使用 Class 对象可能会导致类型擦除问题,从而无法正确地反序列化。例如,我们有一个泛型类 Map<String, List<String>>,如果我们直接使用 Class<Map<String, List<String>>> 来进行反序列化,会因为类型擦除而无法正确地获取到具体的泛型类型参数。

通过 TypeReference,我们可以正确地指定泛型类型参数,从而能够正确地反序列化复杂的泛型类型。

处理复杂泛型的反射操作

在进行反射操作时,我们可能需要获取泛型类型的类型参数。由于类型擦除,直接使用 Class 对象无法获取到这些类型参数。而 TypeReference 可以帮助我们保留这些类型参数,从而能够正确地进行反射操作。

具体使用流程

引入 Jackson 依赖

在使用 TypeReference 之前,我们需要先引入 Jackson 的依赖。如果你使用的是 Maven 项目,可以在 pom.xml 文件中添加以下依赖:

xml 复制代码
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.13.3</version>
</dependency>

定义 TypeReference 的子类

为了使用 TypeReference,我们需要定义一个 TypeReference 的子类。这个子类需要继承自 TypeReference,并且指定具体的泛型类型。例如,如果我们需要处理 Map<String, List<String>> 类型,可以这样定义:

java 复制代码
new TypeReference<Map<String, List<String>>>() {};

使用 TypeReference 进行 JSON 反序列化

在使用 Jackson 进行 JSON 反序列化时,我们可以使用 TypeReference 来指定具体的泛型类型。例如:

java 复制代码
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;

import java.io.IOException;
import java.util.List;
import java.util.Map;

public class TypeReferenceExample {
    public static void main(String[] args) throws IOException {
        String json = "{\"key1\":[\"value1\",\"value2\"],\"key2\":[\"value3\",\"value4\"]}";
        ObjectMapper objectMapper = new ObjectMapper();
        TypeReference<Map<String, List<String>>> typeReference = new TypeReference<Map<String, List<String>>>() {};
        Map<String, List<String>> map = objectMapper.readValue(json, typeReference);
        System.out.println(map);
    }
}

在上面的代码中,我们定义了一个 TypeReference<Map<String, List<String>>> 的子类,并将其传递给 ObjectMapperreadValue 方法。这样,Jackson 就可以正确地反序列化 JSON 字符串到 Map<String, List<String>> 类型。

使用 TypeReference 进行反射操作

在进行反射操作时,我们可以通过 TypeReference 获取泛型类型的类型参数。例如:

java 复制代码
import com.fasterxml.jackson.core.type.TypeReference;

import java.lang.reflect.Type;

public class TypeReferenceReflectionExample {
    public static void main(String[] args) {
        TypeReference<Map<String, List<String>>> typeReference = new TypeReference<Map<String, List<String>>>() {};
        Type type = typeReference.getType();
        System.out.println(type);
    }
}

在上面的代码中,我们通过 TypeReferencegetType 方法获取了泛型类型的 Type 对象。这个 Type 对象包含了泛型类型的类型参数信息,我们可以使用它来进行反射操作。

实际例子:HttpClient 通用类实现

在实际开发中,我们经常需要与外部服务进行 HTTP 通信。为了提高代码的复用性和可维护性,我们通常会自定义一个 HttpClient 类,用于封装 HTTP 请求和响应处理。由于返回值类型不固定,我们可以使用 TypeReference 来实现一个通用的 HttpClient 类。

定义 HttpClient 类

我们定义一个通用的 HttpClient 类,用于发送 HTTP 请求并处理响应。这个类将使用 TypeReference 来支持返回值类型不固定的场景。

java 复制代码
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.util.EntityUtils;

import java.io.IOException;

public class HttpClient {
    private static final ObjectMapper objectMapper = new ObjectMapper();
    private static final CloseableHttpClient httpClient = HttpClients.createDefault();

    public static <T> T get(String url, TypeReference<T> typeReference) throws IOException {
        HttpGet httpGet = new HttpGet(url);
        try (CloseableHttpResponse response = httpClient.execute(httpGet)) {
            String jsonResponse = EntityUtils.toString(response.getEntity());
            return objectMapper.readValue(jsonResponse, typeReference);
        }
    }
}

使用 HttpClient 类

我们可以使用 HttpClient 类来发送 HTTP 请求,并指定返回值的类型。例如,假设我们有一个外部服务返回一个 Map<String, List<String>> 类型的 JSON 数据,我们可以这样使用 HttpClient 类:

java 复制代码
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;

import java.util.List;
import java.util.Map;

public class HttpClientExample {
    public static void main(String[] args) {
        try {
            String url = "https://api.example.com/data";
            Map<String, List<String>> result = HttpClient.get(url, new CustomTypeReference().mapReference);
            System.out.println(result);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

自定义 TypeReference

java 复制代码
import java.util.List;
import java.util.Map;

import com.fasterxml.jackson.core.type.TypeReference;

/**
 * CustomTypeReference 自定义TypeReference
 *
 * @author xxx
 * @since 2025/7/24
 */
public class CustomTypeReference {
    public TypeReference<Map<String, List<String>>> mapReference =
            new TypeReference<Map<String, List<String>>>() {};
}

在上面的代码中,我们定义了一个 TypeReference<Map<String, List<String>>> 的子类,并将其传递给 HttpClient.get 方法。这样,HttpClient 类就可以正确地反序列化返回的 JSON 数据到 Map<String, List<String>> 类型。


至此,本次分享到此结束啦!!!

相关推荐
二哈赛车手5 分钟前
新人笔记---实现简易版的rag的bm25检索(利用ES),以及RAG上传时的ES与向量数据库双写
java·数据库·笔记·spring·elasticsearch·ai
winner88818 分钟前
从零吃透C++命名空间、std、#include、string、vector
java·开发语言·c++
AI人工智能+电脑小能手16 分钟前
【大白话说Java面试题】【Java基础篇】第26题:Java的抽象类和接口有哪些区别
java·开发语言·面试
bzmK1DTbd25 分钟前
SOLID原则在Java中的实践:单一职责与开闭原则
java·开发语言·开闭原则
AI进化营-智能译站28 分钟前
ROS2 C++开发系列07-高效构建机器人决策逻辑,运算符与控制流实战
开发语言·c++·ai·机器人
winner888130 分钟前
C++ 命名空间、虚函数、抽象类、protected 权限全套通俗易懂精讲(附与 Java 对比)
java·开发语言·c++
不会编程的懒洋洋40 分钟前
C# P/Invoke 基础
开发语言·c++·笔记·安全·机器学习·c#·p/invoke
直奔標竿41 分钟前
Java开发者AI转型第二十五课!Spring AI 个人知识库实战(四)——RAG来源追溯落地,拒绝AI幻觉
java·开发语言·人工智能·spring boot·后端·spring
时空系1 小时前
认识Rust——我的第一个程序 Rust中文编程
开发语言·后端·rust
yqcoder1 小时前
JavaScript 柯里化:把“大餐”拆成“小炒”的艺术
开发语言·javascript·ecmascript