美文网首页
Java--反射机制(三)——泛型使用

Java--反射机制(三)——泛型使用

作者: 年少懵懂丶流年梦 | 来源:发表于2017-06-17 09:37 被阅读541次

一、泛型和Class类

从JDK 1.5 后,Java中引入泛型机制,Class 类也增加了泛型功能,从而允许使用泛型来限制Class类。
例如:String.class 的类型实际上是Class<String>。如果Class对应的类暂时未知,则使用Class<?>(?是通配符)。通过在反射中使用泛型,可以避免使用反射生成的对象需要强制类型转换

泛型的好处众多,最主要的一点就是避免类型转换,防止出现ClassCastException,即类型转换异常。以下面程序为例:

public class ObjectFactory {
    public static Object getInstance(String name){
        try {
            //创建指定类对应的Class对象
            Class cls = Class.forName(name);
            //返回使用该Class对象创建的实例
            return cls.newInstance();
        } catch (ClassNotFoundException | InstantiationException | IllegalAccessException e) {
            e.printStackTrace();
            return null;
        }
    }

}

上面程序是个工厂类,通过指定的字符串创建Class对象并创建一个类的实例对象返回。但是这个对象的类型是Object对象,取出实例后需要强制类型转换。

如下例:

Date date = (Date) ObjectFactory.getInstance("java.util.Date");

或者如下:

String string = (String) ObjectFactory.getInstance("java.util.Date");

上面代码在编译时不会有任何问题,但是运行时将抛出ClassCastException异常,因为程序试图将一个Date对象转换成String对象。

但是泛型的出现后,就可以避免这种情况。

public class ObjectFactory {
    public static <T> T getInstance(Class<T> cls) {
        try {
            // 返回使用该Class对象创建的实例
            return cls.newInstance();
        } catch (InstantiationException | IllegalAccessException e) {
            e.printStackTrace();
            return null;
        }
    }

}

在上面程序的getInstance()方法中传入一个 Class<T> 参数,这是一个泛型化的
Class 对象,调用该Class对象的newInstance()方法将返回一个 T 对象。

String instance = ObjectFactory.getInstance(String.class);

通过传入String.class便知道T代表String,所以返回的对象是String类型的,避免强制类型转换。

当然Class类引入泛型的好处不止这一点,在以后的实际应用中会更加能体会到。

二、使用反射来获取泛型信息

通过指定类对应的 Class 对象,可以获得该类里包含的所有 Field,不管该 Field 是使用 private 修饰,还是使用 public 修饰。获得了 Field 对象后,就可以很容易地获得该 Field 的数据类型。

// 获取 Field 对象 f 的类型
Class<?> a = f.getType();

但这种方式只对普通类型的 Field 有效。如果该 Field 的类型是有泛型限制的类型,如 Map<String, Integer> 类型,则不能准确地得到该 Field 的泛型参数。

为了获得指定 Field 的泛型类型,应先使用如下方法来获取指定 Field 的类型。

// 获得 Field 实例的泛型类型
Type type = f.getGenericType();

然后将 Type 对象强制类型转换为 ParameterizedType 对象。
ParameterizedType 代表被参数化的类型,也就是增加了泛型限制的类型。

ParameterizedType 类提供了如下两个方法。

getRawType():返回没有泛型信息的原始类型。

getActualTypeArguments():返回泛型参数的类型。

下面是一个获取泛型类型的完整程序。

public class GenericTest
{
    private Map<String , Integer> score;
    public static void main(String[] args)
        throws Exception
    {
        Class<GenericTest> clazz = GenericTest.class;
        Field f = clazz.getDeclaredField("score");
        // 直接使用getType()取出Field类型只对普通类型的Field有效
        Class<?> a = f.getType();
        // 下面将看到仅输出java.util.Map
        System.out.println("score的类型是:" + a);
        // 获得Field实例f的泛型类型
        Type gType = f.getGenericType();
        // 如果gType类型是ParameterizedType对象
        if(gType instanceof ParameterizedType)
        {
            // 强制类型转换
            ParameterizedType pType = (ParameterizedType)gType;
            // 获取原始类型
            Type rType = pType.getRawType();
            System.out.println("原始类型是:" + rType);
            // 取得泛型类型的泛型参数
            Type[] tArgs = pType.getActualTypeArguments();
            System.out.println("泛型类型是:");
            for (int i = 0; i < tArgs.length; i++) 
            {
                System.out.println("第" + i + "个泛型类型是:" + tArgs[i]);
            }
        }
        else
        {
            System.out.println("获取泛型类型出错!");
        }
    }
}

输出结果:

score 的类型是: interface java.util.Map
原始类型是: interface java.util.Map
泛型类型是:
第 0 个泛型类型是: class java.lang.String
第 1 个泛型类型是:class java.lang.Integer

从上面的运行结果可以看出,直接使用 Field 的 getType() 方法只能获取普通类型的 Field 的数据类型:对于增加了泛型参数的类型的 Field,应该使用 getGenericType() 方法来取得其类型。

Type 也是 java.lang.reflect 包下的一个接口,该接口代表所有类型的公共高级接口,Class 是 Type 接口的实现类。Type 包括原始类型、参数化类型、数组类型、类型变量和基本类型等。

相关文章

  • Java--反射机制(三)——泛型使用

    一、泛型和Class类 从JDK 1.5 后,Java中引入泛型机制,Class 类也增加了泛型功能,从而允许使用...

  • Java高级特性入门——泛型、反射和注解

    本次的分享主要围绕以下三个方面: 一、泛型介绍 二、反射机制 三、注解的使用 一、泛型介绍 在日常编程的过程中,泛...

  • Java泛型

    本文介绍的知识点 泛型是什么? 泛型的使用在反射中使用泛型在集合类中使用泛型 关于泛型擦除如何理解?如何避免泛型擦...

  • 章节笔记——泛型与反射机制

    泛型 反射机制 泛型:由于类型转换错误,编译时不报错,运行时报错。为了弥补这个问题,用泛型弥补,集合类添加对象不需...

  • 掌握java动态代理及原理有多难?

    前言:使用的jdk是1.7,需要了解反射机制 泛型 字节码登概念! 一、代理模式 代理模式是常用的java设计模...

  • Java反射(三)泛型

    一、泛型和Class类从JDK 1.5 后,Java中引入泛型机制,Class类也增加了泛型功能,从而允许使用泛型...

  • Kotlin-泛型和委托

    泛型 泛型的使用是为了程序有更好的扩展性。泛型类和泛型方法 泛型的高级特性java的泛型是通过类型擦除机制来实现的...

  • Java反射机制总结(三)

    一、泛型和Class类 从JDK 1.5 后,Java中引入泛型机制,Class类也增加了泛型功能,从而允许使用泛...

  • 【阿里P8大牛教你Android入门之路(java篇)】Java

    一、泛型和Class类 从JDK 1.5 后,Java中引入泛型机制,Class类也增加了泛型功能,从而允许使用泛...

  • 泛型 & 注解 & Log4J日志组件

    掌握的知识 : 基本用法、泛型擦除、泛型类/泛型方法/泛型接口、泛型关键字、反射泛型(案例) 泛型 概述 : 泛型...

网友评论

      本文标题:Java--反射机制(三)——泛型使用

      本文链接:https://www.haomeiwen.com/subject/rnumqxtx.html