一、什么是反射?

在运行状态中,对于任意一个类,都能够获取到这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性(包括私有的方法和属性),这种动态获取的信息以及动态调用对象的方法的功能就称为java语言的反射机制。通俗点讲,通过反射,该类对我们来说是完全透明的,想要获取任何东西都可以。
想要使用反射机制,就必须要先获取到该类的字节码文件对象(.class),通过字节码文件对象,就能够通过该类中的方法获取到我们想要的所有信息(方法,属性,类名,父类名,实现的所有接口等等),每一个类对应着一个字节码文件也就对应着一个Class类型的对象,也就是字节码文件对象。
Class对象的由来是将class文件读入内存,并为之创建一个Class对象。Java反射 - 图1

1、基本概念:

(1)反射:框架设计的灵魂

1)框架:半成品软件。可以在框架的基础上进行软件开发,简化编码。

2)反射:将类的各个组成部分封装为其他对象,这就是反射机制。

好处:可以在程序运行过程中,操作这些对象。
可以解耦,提高程序的可扩展性。

2、获取字节码文件(class)对象的三种方式。

(1)Class clazz1 = Class.forName(“全限定类名”); //通过Class类中的静态方法forName,直接获取到一个类的字节码文件对象,此时该类还是源文件阶段,并没有变为字节码文件。

多用于配置文件,将类名定义在配置文件中。读取文件,加载类

(2)Class clazz2 = Person.class; //当类被加载成.class文件时,此时Person类变成了.class,在获取该字节码文件对象,也就是获取自己, 该类处于字节码阶段。

多用于参数的传递

(3)Class clazz3 = p.getClass(); //通过类的实例获取该类的字节码文件对象,该类处于创建对象阶段

多用于对象的获取字节码的方式

结论:同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。

3、Class对象功能

获取功能:

(1)获取成员变量们

Field[] getFields()
Field getField(String name)

Field[] getDeclaredFields()
Field getDeclaredFields(String name)

(2)获取构造方法们

Constructor<?> [] getConstructors()
Constructor getConstructors(类<?>… parameterTypes)

Constructor getDeclaredConstructors(类<?>… parameterTypes)
Constructor<?>[] getDeclaredConstructors()

(3)获取成员方法们:

Method[] getMethods()
Method getMethod(String name,类<?>… paramterTypes)

Method[] getDeclaredMethods()
Method getDeclaredMethod(String name,类<?>… paramterTypes)

(4)获取类名

*String getName()

image.png
image.png
image.png

image.png

image.png
image.png
image.png
有了字节码文件对象才能获得类中所有的信息,我们在使用反射获取信息时,也要考虑使用上面哪种方式获取字节码对象合理,视不同情况而定。下面介绍Class类的功能

  1. Class: 是一个类; 一个描述类的类.
      封装了描述方法的 Method,
         描述字段的 Filed,
    描述构造器的 Constructor 等属性.

    2. 如何得到 Class 对象:
      2.1 Person.class
      2.2 person.getClass()
      2.3 Class.forName(“com.atguigu.javase.Person”)

    3. 关于 Method:
      3.1 如何获取 Method:
        1). getDeclaredMethods: 得到 Method 的数组.
        2). getDeclaredMethod(String methondName, Class … parameterTypes)

      3.2 如何调用 Method
        1). 如果方法时 private 修饰的, 需要先调用 Method 的 setAccessible(true), 使其变为可访问
        2). method.invoke(obj, Object … args);

    4. 关于 Field:
      4.1 如何获取 Field: getField(String fieldName)
      4.2 如何获取 Field 的值:
        1). setAccessible(true)
        2). field.get(Object obj)
      4.3 如何设置 Field 的值:
        field.set(Obejct obj, Object val)

    5. 了解 Constructor 和 Annotation

    6. 反射和泛型.
      6.1 getGenericSuperClass: 获取带泛型参数的父类, 返回值为: BaseDao
      6.2 Type 的子接口: ParameterizedType
      6.3 可以调用 ParameterizedType 的 Type[] getActualTypeArguments() 获取泛型参数的数组.
    image.png
    image.png
    image.png
    image.png