Java泛型T的应用 获取T类型

本文深入探讨了Java中泛型获取机制,特别是如何在内部匿名类中获取T类型的实例,并通过示例代码揭示了可能出现的Class无法强转为ParameterizedType异常的原因及解决方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Java中 获取 T 类型 需要注意T类型只能在继承类里获取 比如

import java.lang.reflect.ParameterizedType;
        import java.lang.reflect.Type;

public class Call<T> {
    private Object obj;

    public Call(Object obj) {
        this.obj = obj;
    }

    public T getT() {
        System.out.println(getTClass().getName());
        if (obj.getClass().getName().equals(getTClass().getName())) {
            return ((T) obj);
        }
        return null;
    }

    private Class<T> getTClass() {
        Type t = getClass().getGenericSuperclass();
        Type[] params = ((ParameterizedType) t).getActualTypeArguments();
        Class<T> cls = (Class<T>) params[0];

        return cls;
    }

}

此时

String str = new Call(“Hello”).getT();
print(str);

则会报异常
异常信息为

Class无法强转为ParameterizedType

是因为泛型只能在继承类中获取 加个继承类

public class StringCall extends Call {

    public StringCall(Object obj) {
        super(obj);
    }

}

此时

String str = new StringCall(“Hello”).getT();
print(str);

输出正常

Hello

此时

String str = new StringCall(“Hello”){}.getT();
print(str);

则会报异常
异常信息为

Class无法强转为ParameterizedType

虚拟机无法获取一个内部匿名类的T类型
Java机制 内部匿名类 传递的是当前类的类型 如下

public class Main {
  public static void main(String[] args) {
     String str = new StringCall(“Hello”){}.getT();
     print(str);
  }
}

StringCall的getT方法中 得到的类型值是

Main$1
没有T类型

以上

### Java `<T>` 的用法 #### 的概念 Java 中的是一种允许开发者编写可以处理多种数据类型的通用算法的技术。它提供了更高的灵活性和更强的类型安全性,使得编译器能够在编译期捕获潜在的类型错误。 #### 类级别的 当在一个类中声明参数时,可以通过在类名后面加尖括号来定义类型 `T`。例如: ```java public class Box<T> { private T content; public void setContent(T content) { this.content = content; } public T getContent() { return content; } } ``` 在这个例子中,`Box` 是一个容器类,它可以存储任何类型的对象[^1]。通过这种方式,同一个类可以在不同的上下文中用于不同类型的对象,而无需为每种类型创建单独的类版本。 #### 方法级别的 除了在类级别上使用外,也可以在方法层面应用。这意味着即使是在非类中,也能够定义支持的方法。下面是一个简单的方法的例子: ```java public static <E> void printArray(E[] elements){ for (E element : elements){ System.out.println(element); } } ``` 这里展示了如何利用让单个方法适应各种数组类型的数据打印需求[^2]。 #### 关于伪的理解 值得注意的是,在 Java 编译过程中会执行一种称为 **类型擦除** 的操作。这种机制意味着所有的信息仅存在于源码阶段;一旦经过编译,实际生成的字节码将不再保留这些具体的信息。因此,尽管表面上看似乎实现了真正的功能,但实际上由于运行时刻无法获取确切的类型信息,所以称之为“伪”。这一特性带来了一些限制,比如不能实例化变量或者创建带有原始类型作为形参的新对象等情形[^3]。 #### 实际应用场景举例 考虑这样一个场景——我们需要设计一个工具函数用来交换两个相同类型的值的位置。借助于技术,我们可以轻松完成如下实现: ```java public class Util{ public static <K> void swap(K[] array, int i, int j){ K temp = array[i]; array[i] = array[j]; array[j] = temp; } } // 使用示例 Integer[] numbers={1, 2}; Util.swap(numbers, 0, 1); System.out.println(Arrays.toString(numbers)); // 输出 "[2, 1]" ``` 上述代码片段清晰地展现了带来的便利性和可读性提升效果[^4]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值