Java反射---方法的反射、深入了解泛型

简介: 版权声明:本文为博主原创文章,转载请注明出处。 https://blog.
版权声明:本文为博主原创文章,转载请注明出处。 https://blog.csdn.net/twilight_karl/article/details/73848996

方法的反射

invoke(对象,参数列表)
1、获得类类型
Class data = a.getClass();

2、获得方法
- getMethod(“方法名”,可变参数(Class[])) 获得自身和继承的public方法
- getDeclaredMethod(“方法名”,可变参数(Class[])) 获得自身的所有方法,不包括继承的方法

    Method print = data.getMethod("print", new Class[]{int.class,int.class});
    Method print2 = data.getDeclaredMethod("print", new Class[]{String.class,String.class});

3、执行方法
invoke(对象obj,参数列表) 使用对象obj调用方法

        print.invoke(a, 10, 20);
        print2.invoke(a, new String[]{"hello","world"});

如果没有参数,可以不写:

        Method print3 = data.getMethod("print");
        print3.invoke(a);

通过反射了解泛型的本质

所谓泛型,是在编译阶段判断变量类型是否满足泛型要求。例如ArrayList<Stirng> 在编译时如果add一个int类型的变量肯定会出错,但是,在编译完成后,不同类型的泛型实际上是一样的。如下:

        ArrayList list1 = new ArrayList();
        ArrayList<String> list2 = new ArrayList<String>();

        Class c1 = list1.getClass();
        Class c2 = list2.getClass();

        System.out.println(c1 == c2); // true

说明不同类型的泛型在编译后是一样的,泛型只是帮助判断变量类型的一种机制。
String类型的泛型,只能加入String类型的变量,所以集合中保存的变量原本都是String类型。而Object类型的泛型可以加入任意类型的变量,所有变量都会被转换为Object类型后保存,所以在去变量时,需要手动的将Object类型强制转换为指定的类型。

反射机制是在编译之后完成的。所以可以利用反绕过泛型的编译:

        Method add = c2.getMethod("add", Object.class);
        add.invoke(list2, 100);

        System.out.println(list2.size());       // 1
        System.out.println(list2);              // [100]

上面的例子在String类型的泛型集合中加入了整形变量,说明泛型的类型检查是在运行之前进行的。编译过后不会有泛型的类型检查,所以不会报错。另外我在编写的过程中还发现一个细节:

        Method add = c2.getMethod("add", String.class);

上面这行代码会报错,c2是ArrayList<String>的类类型。一个String类型的泛型集合却无法获得String类型的add方法。原因是在编译过后所有类型会被擦除。在ArrayList的源码中用一个Object类型的数组来储存数据。编译过变量会被保存在Object数组中,所以add方法中的类型也会被转化为Object

    transient Object[] elementData; // non-private to simplify nested class access

个人理解,欢迎指正

参考资料:反射

相关文章
|
8天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
41 4
|
19天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
39 17
|
13天前
|
Java 测试技术 Maven
Java一分钟之-PowerMock:静态方法与私有方法测试
通过本文的详细介绍,您可以使用PowerMock轻松地测试Java代码中的静态方法和私有方法。PowerMock通过扩展Mockito,提供了强大的功能,帮助开发者在复杂的测试场景中保持高效和准确的单元测试。希望本文对您的Java单元测试有所帮助。
18 2
|
20天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
15 3
|
20天前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
14 2
|
19天前
|
Java API
[Java]泛型
本文详细介绍了Java泛型的相关概念和使用方法,包括类型判断、继承泛型类或实现泛型接口、泛型通配符、泛型方法、泛型上下边界、静态方法中使用泛型等内容。作者通过多个示例和测试代码,深入浅出地解释了泛型的原理和应用场景,帮助读者更好地理解和掌握Java泛型的使用技巧。文章还探讨了一些常见的疑惑和误区,如泛型擦除和基本数据类型数组的使用限制。最后,作者强调了泛型在实际开发中的重要性和应用价值。
14 0
[Java]泛型
|
20天前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
14 1
|
20天前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
27 1
|
20天前
|
Java
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅。它们用于线程间通信,使线程能够协作完成任务。通过这些方法,生产者和消费者线程可以高效地管理共享资源,确保程序的有序运行。正确使用这些方法需要遵循同步规则,避免虚假唤醒等问题。示例代码展示了如何在生产者-消费者模型中使用`wait()`和`notify()`。
22 1
|
20天前
|
存储 Java
[Java]反射
本文详细介绍了Java反射机制的基本概念、使用方法及其注意事项。首先解释了反射的定义和类加载过程,接着通过具体示例展示了如何使用反射获取和操作类的构造方法、方法和变量。文章还讨论了反射在类加载、内部类、父类成员访问等方面的特殊行为,并提供了通过反射跳过泛型检查的示例。最后,简要介绍了字面量和符号引用的概念。全文旨在帮助读者深入理解反射机制及其应用场景。
13 0
[Java]反射