【Java进阶篇】——反射机制

一、反射的概念

1.1 反射出现的背景

  • Java程序中,所有对象都有两种类型:编译时类型和运行时类型,而很多时候对象的编译时类型和运行时类型不一致
    Object obj = new String("hello")obj.getClass();

如果某些变量或形参的声明类型是Object类型,但程序却需要调用该对象运行时类型的方法,而不是编译时中的方法,那么如何解决呢?

  • 方案一:在编译和运行时都完全知道类型的具体信息,在这种情况下,我们可以直接使用 instanceof 运算符进行判断,再利用强制类型转换符将其转换成运行时类型的变量即可

    • instanceof 是用来判断左边的对象是否为它右边类的实例 num instanceof int
  • 方案二:编译时根本无法预知该对象和类的真实信息,程序只能依靠运行时信息来发现该对象和类的真实信息,这种情况就必须使用反射机制

1.2 反射概述

  • 反射(Reflection):被视为动态语言的关键,反射机制允许程序在运行期间借助于Reflection API 取得任何类的内部信息,并能直接操作任意对象的内部属性及方法

  • 加载完类之后,在堆内存的方法区中就产生了一个Class类型的对象【一个类只有一个Class对象】

    • 这个对象包含完整的类的结构信息
      请添加图片描述

    • 在内存加载的过程中,可以参考下面的图
      请添加图片描述

1.3 反射机制提供的功能

  • 在运行时判断任意一个对象所属类
  • 在运行时构造任意一个类的对象
  • 在运行时判断任意一个类所具有的成员和方法
  • 在运行时获取泛型信息
  • 在运行时调用任意一个对象的成员变量和方法
  • 在运行时处理注解
  • 生成动态代理

1.4 反射相关的API

  • java.lang.Class: 代表一个类
    • java.lang.reflect.Method: 代表类的方法
    • java.lang.reflect.Field: 代表类的成员变量
    • java.lang.reflect.Constructor: 代表类的构造器

1.5 反射机制的优缺点

  • 优点:

    • 提高了Java程序的灵活性和扩展性,降低了耦合度,提高了自适应能力
    • 允许程序创建和控制任何类的对象,无需提前硬编码目标类
  • 缺点:

    • 反射的性能较低
      • 反射机制主要应用在堆灵活性和扩展性要求较高的系统框架上
    • 反射会模糊程序内部逻辑,可读性较差

二、理解Class类并获取Class实例

  • 要想要深度剖析一个类,首先要获得这个类的Class对象,然后调用相应的API分析
    • java.lang.Class、java.lang.reflect.* >> Class对象是反射的根源

2.1 理解Class类

🌔 1、在理论上我们如何理解Class呢?

  • 在Object类中定义了以下的方法,此方法将被所有子类继承:
    public final Class getClass()
    • 我们可以看出这个方法的返回值是一个Class类,这个类是Java反射的源头,可以通过对象反射得到类的名称

请添加图片描述

- 对于每个类而言,JRR都为其保留一个不变的Class类型的对象,其中包含了特定某个结构的信息
	- Class 本身也是一个类
	- Class 对象只能由系统建立对象
	- 一个加载的类在JVM中只会有一个Class实例
	- 一个Class对象对应的是 ==一个加载到JVM中的一个.class文件==
	-  每个类的实例都会记得自己是由哪个Class实例所生成
	- 通过Class可以完整地得到一个类中的所有被加载的结构
	- Class类是Reflection的根源,针对任何想要动态加载、运行的类,只能首先获得相应的Class对象

🌔 2、在内存结构上如何理解Class?

请添加图片描述
说明:上图中字符串常量池在JDK6中存储在方法区;JDK7及以后,存储在堆空间

2.2 获取Class类实例的方法

🌔 1、编译期间已知类型

  • 如果知道具体的类,我们可以直接通过该类的 class 属性获取,该方法安全可靠、程序性能最高
    Class clazz = String.class;

🌔 2、获取对象的运行时类型

  • 已知某个类的实例,调用该实例的getClass()方法获取Class对象

    Person p = new Person();
    Class clazz = p.getClass();
    

🌔 3、可以获取编译期间的未知类型

  • 已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,可能抛出 ClassNotFoundException

    Class clazz = Class.forName("java.lang.String");

🌔 4、可以使用加载类的方式

  • 用系统类加载对象或自定义加载器对象加载指定路径下的类型

    ClassLoader c1 = this.getClass().getClassLoader(); Class clazz = c1.loadClass("类的全限定类名");

🌔 5、通过综合的案例演示如何获取Class实例

package com.zwh.shangguigu.reflection_;

/**
 * @author Bonbons
 * @version 1.0
 */
public class GetClassObject {
    public static void main(String[] args) {
        //知道编译时类型
        Class c1 = GetClassObject.class;
        //获取对象的运行时类型
        GetClassObject obj = new GetClassObject();
        Class c2 = obj.getClass();
        Class c3 = null;
        Class c4 = null;
        try{
            //通过指定的类名去查找、创建Class的实例
            c3 = Class.forName("com.zwh.shangguigu.reflection_.GetClassObject");
            //通过类加载器去获得Class的实例
            c4 = ClassLoader.getSystemClassLoader().loadClass("com.zwh.shangguigu.reflection_.GetClassObject");
        }catch (ClassNotFoundException e){
            e.printStackTrace();
        }

        //输出四个Class实例
        System.out.println("c1 = " + c1);
        System.out.println("c2 = " + c2);
        System.out.println("c3 = " + c3);
        System.out.println("c4 = " + c4);

        //通过Class.class方法获取到的最可靠,所以用它与其他几个比较
        System.out.println(c1 == c2);
        System.out.println(c1 == c3);
        System.out.println(c1 == c4);
    }
}

在这里插入图片描述

2.3 哪些类型可以有Class对象

  • Java中的所有类型都可以用Class对象 【这个Class对象就是类实例创建后在JVM中的.class文件】

在这里插入图片描述

Class c1 = Object.class;
Class c2 = Comparable.class;
Class c3 = String[].class;
Class c4 = int[][].class;
Class c5 = ElementType.class;
Class c6 = Override.class;
Class c7 = int.class;
Class c8 = void.class;
Class c9 = Class.class;

int[] a = new int[10];
int[] b = new int[100];
Class c10 = a.getClass();
Class c11 = b.getClass();
// 只要元素类型与维度一样,就是同一个Class
System.out.println(c10 == c11);

2.4 Class类的常用方法

方法名功能
static Class forName(String name)返回指定类名name的Class对象
Object newInstance()调用缺省构造函数,返回该Class对象的一个实例
getName()返回此Class对象所表示的实体名称【类、接口、数组等】
Class getSuperClass()返回当前Class对象的父类的Class对象
Class [] getInterfaces()获取当前Class对象的接口
ClassLoader getClassLoader()返回该类的类加载器
Class getSuperclass()返回表示此Class所表示实体的超类的Class
Constructor [] getConstructors()返回一个包含某个Constructor对象的数组
Field[] getDeclaredFields()返回Field对象的一个数组
Method getMethod(String name, Class …paramTypes)返回一个Method对象,此对象的参数类型为paramType
//test4.Person是test4包下的Person类
String str = "test4.Person";
//获取这个类的Class实例
Class clazz = Class.forName(str);
//根据Class创建对象
Object obj = clazz.newInstance();
//获取特定的一个属性
Field field = clazz.getField("name");
//将obj这个对象的这个name属性设置为 "Peter"
field.set(obj, "Peter");
//获取obj的name属性值
Object name = field.get(obj);
//输出name属性值,看是否设定成功
System.out.println(name);

三、类的加载与ClassLoader的理解

3.1 类的生命周期

  • 类在内存中完整的生命周期:加载 >> 使用 >> 卸载
    • 加载:装载 >> 链接 >> 初始化三个阶段

请添加图片描述

3.2 类的加载过程

  • 当程序主动使用某个类时,如果该类还未被加载到内存中,系统会通过加载、链接初始化三个步骤对该类进行初始化【统称类加载 】

请添加图片描述

  • 我们展开论述一下类加载包含的三个子过程
    • 装载(Loading):类加载器将类的class文件读入内存,并为之创建一个 java.lang.Class对象
    • 链接(Linking):
      • 验证(Verify):确保加载的类信息符合JVM规范
      • 准备 (Prepare):正式为类变量(static)分配内存并设置类变量默认初始值阶段,这些内存都将在方法区中进行分配
      • 解析(Resolve):虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程
    • 初始化(Initization):
      • 执行类构造器<clinit>() 方法的过程
        • 类构造器<clinit>()方法:是由编译期自动收集类中所有类变量的赋值动作静态代码块中的语句合并产生的
        • 类构造器是构造类信息的,不是构造该类对象的构造器
      • 当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化
      • 虚拟机会保证一个类的<clinit>()方法,在多线程环境中被正确加锁和同步

3.3 类加载器

🌔 1、类加载器有什么作用呢?

  • 将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的 java.lang.Class 对象,作为方法区中类数据的访问入口

请添加图片描述

  • 类缓存:标准的JavaSE类加载器可以按要求查找类,但一旦某个类被加载到类加载器中,它将维持加载一段时间(缓存),JVM的垃圾回收机制可以回收这些Class对象

🌔 2、以JDK8为例介绍类加载器的分类

  • JVM 支持两种类型的类加载器:

    • 引导类加载器(Bootstrap ClassLoader)
    • 自定义类加载器(User-Defined ClassLoader):将所有派生于抽象类ClassLoader的类加载器都划分为自定义类加载器
  • 常见的类加载器结构如图所示:

请添加图片描述
(1)启动类加载器(引导类加载器,Bootstrap ClassLoader)

  • 这个类加载使用 C/C++ 语言实现的,嵌套在JVM内部,获取他的对象往往返回 null
  • 它用来加载Java的核心库,用于提供JVM自身需要的类
    • JAVA_HOME/jre/lib/rt.jar或sun.boot.class.path路径下的内容
  • 并不继承在 java.lang.ClassLoader,没有父加载器
  • 出于安全考虑,Bootstrap启动类加载器只加载包名为 java、javax、sun开头的类
  • 是加载扩展类和应用程序类加载器的父类加载器

(2)扩展类加载器(Extension ClassLoader)

  • Java语言编写,由 sun.misc.Launcher&ExtClassLoader实现
  • 继承与ClassLoader类,父加载器是启动类加载器
  • 从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录的 jre/lib/ext 子目录下加载类库
    • 如果用户创建的JRE放在此目录下,也会自动由扩展类加载器加载

请添加图片描述

(3)应用程序类加载器(系统类加载器, AppClassLoader)

  • Java语言编写,由 sun.misc.Luncher&AppClassLoader实现
  • 继承与ClassLoader类,是扩展类加载器的子类
  • 它负责加载环境变量 classpath 或系统属性 java.class.path指定路径下的类库
  • 应用程序中的类加载器默认是系统类加载器
  • 它是用户自定义类加载器的默认父加载器
  • 通过ClassLoader的getSystemClassLoader()方法可以获取到该类加载器

(4)用户自定义类加载器(了解)

  • 在Java日常应用程序开发中,类的加载几乎是由上述3中类加载器互相配合执行的。必要情况下我们可以自定义类加载器,来定义类的加载方式
  • 可以实现类库的动态加载,加载源可以是本地的JAR包,也可以是网络上的远程资源
  • 可以实现应用隔离(隔离不同的组件模块),通常需要继承ClassLoader

🌔 3、如何查看某个类的类加载器对象呢?

获取默认的系统类加载器

ClassLoader classloader = ClassLoader.getSystemClassLoader();

查看某个类时哪个类加载器加载的

ClassLoader classloader = Class.forName("exer2.ClassloaderDemo").getClassLoader();
//如果是根加载器加载的类,得到的时null
ClassLoader classloader1 = Class.forName("java.lang.Object).getClassLoader();

获取某个类加载器的父加载器

ClassLoader parneClassloader = classloader.getParent();
  • 通过代码演示如何获取加载器:
package com.zwh.shangguigu.reflection_;

/**
 * @author Bonbons
 * @version 1.0
 */
public class ClassLoaderTest {
    public static void main(String[] args) {
        //获取默认系统类加载器
        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
        System.out.println("默认的系统类加载器: " + systemClassLoader);
        //String类加载器
        ClassLoader c1 = String.class.getClassLoader();
        System.out.println("加载String类的类加载器: " + c1);
        //通过forName获取Class实例,然后进一步获取其类加载器
        try{
            ClassLoader c2 = Class.forName("sun.util.resources.cldr.zh.TimeZoneNames_zh").getClassLoader();
            System.out.println("sun.util.resources.cldr.zh.TimeZoneNames_zh类的类加载器: " + c2);
        }catch (ClassNotFoundException e){
            e.printStackTrace();
        }
        //获取当前类的类加载器
        ClassLoader c3 = ClassLoaderTest.class.getClassLoader();
        System.out.println("加载当前类的类加载器: " + c3);
        //当前类加载器的父类的类加载器
        ClassLoader c4 = c3.getParent();
        System.out.println("当前类的父类的类加载器: " + c4);
        //当前类父类的父类
        ClassLoader c5 = c4.getParent();
        System.out.println("当前类父类的父类的类加载器: " + c5);
    }
}

在这里插入图片描述
🌔 4、如何使用ClassLoader获取流呢?

  • getResourceAsStream(String str):获取类路径下的指定文件的输入流
    InputStream in = null;
    in = this.getClass().getClassLoader().getResourceAsStream("exer2\\test.properties");
    System.out.println(in);
    

通过代码演示类加载器获取流:

package com.zwh.shangguigu.reflection_;

import jdk.internal.util.xml.impl.Input;

import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;

/**
 * @author Bonbons
 * @version 1.0
 */
public class ClassLoaderGetStreamTest {
    public static void main(String[] args) throws IOException {
        //创建保存流的集合
        Properties pros = new Properties();

        //方法1:此时默认的相对路径是当前的module
        FileInputStream is = new FileInputStream("info.properties");
        FileInputStream is2 = new FileInputStream("src//info1.properties");

        //方法2:使用类的加载器,此时默认的相对路径是当前module的src目录
        InputStream is3 = ClassLoader.getSystemClassLoader().getResourceAsStream("info1.properties");

        pros.load(is);
        pros.load(is2);
        pros.load(is3);

        //获取配置文件中的信息
        String name = pros.getProperty("name");
        String password = pros.getProperty("password");
        System.out.println("name = " + name + ", password = " + password);
    }
}

四、反射的基本应用

4.1 创建运行时类的对象

🌔 1、直接调用Class对象的newInstance() 方法

  • 要求:
    • 必须有一个无参数的构造器
    • 类构造器的访问权限足够
  • 步骤:
    • 获取该类型的Class对象
    • 调用Class对象的newInstance()方法创建对象

🌔 2、获取构造器对象来进行实例化

  • 步骤:

    • 通过Class类的getDeclaredConstructor(Class parameterTypes) 取得本类的指定形参类型的构造器
    • 向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数
    • 通过Constructor实例化对象
  • 如果构造器的权限修饰符修饰的范围不可见,也可以调用setAccessible(true)

package com.atguigu.reflect;

import org.junit.Test;

import java.lang.reflect.Constructor;

public class TestCreateObject {
    @Test
    public void test1() throws Exception{
//        AtGuiguClass obj = new AtGuiguClass();//编译期间无法创建

        Class<?> clazz = Class.forName("com.atguigu.ext.demo.AtGuiguClass");
        //clazz代表com.atguigu.ext.demo.AtGuiguClass类型
        //clazz.newInstance()创建的就是AtGuiguClass的对象
        Object obj = clazz.newInstance();
        System.out.println(obj);
    }

    @Test
    public void test2()throws Exception{
        Class<?> clazz = Class.forName("com.atguigu.ext.demo.AtGuiguDemo");
        //java.lang.InstantiationException: com.atguigu.ext.demo.AtGuiguDemo
        //Caused by: java.lang.NoSuchMethodException: com.atguigu.ext.demo.AtGuiguDemo.<init>()
        //即说明AtGuiguDemo没有无参构造,就没有无参实例初始化方法<init>
        Object stu = clazz.newInstance();
        System.out.println(stu);
    }

    @Test
    public void test3()throws Exception{
        //(1)获取Class对象
        Class<?> clazz = Class.forName("com.atguigu.ext.demo.AtGuiguDemo");
        /*
         * 获取AtGuiguDemo类型中的有参构造
         * 如果构造器有多个,我们通常是根据形参【类型】列表来获取指定的一个构造器的
         * 例如:public AtGuiguDemo(String title, int num)
         */
        //(2)获取构造器对象
        Constructor<?> constructor = clazz.getDeclaredConstructor(String.class,int.class);

        //(3)创建实例对象
        // T newInstance(Object... initargs)  这个Object...是在创建对象时,给有参构造的实参列表
        Object obj = constructor.newInstance("尚硅谷",2022);
        System.out.println(obj);
    }
}

4.2 获取运行时类的完整结构

  • 可以获取:包、修饰符、类型名、父类(包括泛型父类)、父接口(包含泛型父接口)、成员(属性、构造器、方法)、注解(类上的、方法上的、属性上的)

接下来重点介绍各种API和演示:

🌔 1、相关API

//1.实现的全部接口
public Class<?>[] getInterfaces()   
//确定此对象所表示的类或接口实现的接口。 

//2.所继承的父类
public Class<? Super T> getSuperclass()
//返回表示此 Class 所表示的实体(类、接口、基本类型)的父类的 Class。

//3.全部的构造器
public Constructor<T>[] getConstructors()
//返回此 Class 对象所表示的类的所有public构造方法。
public Constructor<T>[] getDeclaredConstructors()
//返回此 Class 对象表示的类声明的所有构造方法。

//Constructor类中:
//取得修饰符: 
public int getModifiers();
//取得方法名称: 
public String getName();
//取得参数的类型:
public Class<?>[] getParameterTypes();

//4.全部的方法
public Method[] getDeclaredMethods()
//返回此Class对象所表示的类或接口的全部方法
public Method[] getMethods()  
//返回此Class对象所表示的类或接口的public的方法

//Method类中:
public Class<?> getReturnType()
//取得全部的返回值
public Class<?>[] getParameterTypes()
//取得全部的参数
public int getModifiers()
//取得修饰符
public Class<?>[] getExceptionTypes()
//取得异常信息

//5.全部的Field
public Field[] getFields() 
//返回此Class对象所表示的类或接口的public的Field。
public Field[] getDeclaredFields() 
//返回此Class对象所表示的类或接口的全部Field。

//Field方法中:
public int getModifiers()
//以整数形式返回此Field的修饰符
public Class<?> getType()  
//得到Field的属性类型
public String getName()  
//返回Field的名称。

//6. Annotation相关
get Annotation(Class<T> annotationClass) 
getDeclaredAnnotations() 

//7.泛型相关
//获取父类泛型类型:
Type getGenericSuperclass()
//泛型类型:ParameterizedType
//获取实际的泛型类型参数数组:
getActualTypeArguments()

//8.类所在的包
Package getPackage() 

🌔 2、获取所有属性及相关细节

package com.atguigu.java2;

import java.lang.reflect.Field;
import java.lang.reflect.Modifier;

import org.junit.Test;

import com.atguigu.java1.Person;

public class FieldTest {
	
	@Test
	public void test1(){
		
		Class clazz = Person.class;
		//getFields():获取到运行时类本身及其所有的父类中声明为public权限的属性
//		Field[] fields = clazz.getFields();
//
//		for(Field f : fields){
//			System.out.println(f);
//		}
		
		//getDeclaredFields():获取当前运行时类中声明的所有属性
		Field[] declaredFields = clazz.getDeclaredFields();
		for(Field f : declaredFields){
			System.out.println(f);
		}
	}
	
	//权限修饰符  变量类型  变量名
	@Test
	public void test2(){
		Class clazz = Person.class;
        Field[] declaredFields = clazz.getDeclaredFields();
        for(Field f : declaredFields){
            //1.权限修饰符
            /*
         	* 0x是十六进制
         	* PUBLIC           = 0x00000001;  1    1
         	* PRIVATE          = 0x00000002;  2	10
         	* PROTECTED        = 0x00000004;  4	100
         	* STATIC           = 0x00000008;  8	1000
         	* FINAL            = 0x00000010;  16	10000
         	* ...
         	*
         	* 设计的理念,就是用二进制的某一位是1,来代表一种修饰符,整个二进制中只有一位是1,其余都是0
         	*
         	* mod = 17          0x00000011
         	* if ((mod & PUBLIC) != 0)  说明修饰符中有public
         	* if ((mod & FINAL) != 0)   说明修饰符中有final
         	*/
            int modifier = f.getModifiers();
            System.out.print(Modifier.toString(modifier) + "\t");

//            //2.数据类型
            Class type = f.getType();
            System.out.print(type.getName() + "\t");
//
//            //3.变量名
            String fName = f.getName();
            System.out.print(fName);
//
            System.out.println();
        }
	}
}

🌔 3、获取所有的方法及相关细节

package com.atguigu.java2;

import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

import org.junit.Test;

import com.atguigu.java1.Person;

public class MethodTest {

	@Test
	public void test1() {

		Class clazz = Person.class;
		// getMethods():获取到运行时类本身及其所有的父类中声明为public权限的方法
		// Method[] methods = clazz.getMethods();
		//
		// for(Method m : methods){
		// System.out.println(m);
		// }

		// getDeclaredMethods():获取当前运行时类中声明的所有方法
		Method[] declaredMethods = clazz.getDeclaredMethods();
		for (Method m : declaredMethods) {
			System.out.println(m);
		}
		//
	}

	// 注解信息
	// 权限修饰符 返回值类型 方法名(形参类型1 参数1,形参类型2 参数2,...) throws 异常类型1,...{}
	@Test
	public void test2() {
		Class clazz = Person.class;
		Method[] declaredMethods = clazz.getDeclaredMethods();
		for (Method m : declaredMethods) {
			// 1.获取方法声明的注解
			Annotation[] annos = m.getAnnotations();
			for (Annotation a : annos) {
				System.out.println(a);
			}

			// 2.权限修饰符
			System.out.print(Modifier.toString(m.getModifiers()) + "\t");

			// 3.返回值类型
			System.out.print(m.getReturnType().getName() + "\t");

			// 4.方法名
			System.out.print(m.getName());
			System.out.print("(");
			// 5.形参列表
			Class[] parameterTypes = m.getParameterTypes();
			if (!(parameterTypes == null && parameterTypes.length == 0)) {
				for (int i = 0; i < parameterTypes.length; i++) {

					if (i == parameterTypes.length - 1) {
						System.out.print(parameterTypes[i].getName() + " args_" + i);
						break;
					}

					System.out.print(parameterTypes[i].getName() + " args_" + i + ",");
				}
			}

			System.out.print(")");

			// 6.抛出的异常
			Class[] exceptionTypes = m.getExceptionTypes();
			if (exceptionTypes.length > 0) {
				System.out.print("throws ");
				for (int i = 0; i < exceptionTypes.length; i++) {
					if (i == exceptionTypes.length - 1) {
						System.out.print(exceptionTypes[i].getName());
						break;
					}
					System.out.print(exceptionTypes[i].getName() + ",");
				}
			}
			System.out.println();
		}
	}
}

🌔 4、获取其他结构(构造器、父类、接口、包、注解等)

package com.atguigu.java2;

import com.atguigu.java1.Person;
import org.junit.Test;

import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;

/**
 * @author 尚硅谷-宋红康
 * @create 2020 下午 2:47
 */
public class OtherTest {

    /*
    	获取当前类中的所有的构造器
     */
    @Test
    public void test1(){
        Class clazz = Person.class;
        Constructor[] cons = clazz.getDeclaredConstructors();
        for(Constructor c :cons){
            System.out.println(c);
        }
    }
    /*
    	获取运行时类的父类
     */
    @Test
    public void test2(){
        Class clazz = Person.class;
        Class superclass = clazz.getSuperclass();
        System.out.println(superclass);//class com.atguigu.java1.Creature
    }
    /*
    	获取运行时类的所在的包
     */
    @Test
    public void test3(){
        Class clazz = Person.class;
        Package pack = clazz.getPackage();
        System.out.println(pack);

    }
    /*
    	获取运行时类的注解
     */
    @Test
    public void test4(){
        Class clazz = Person.class;
        Annotation[] annos = clazz.getAnnotations();
        for (Annotation anno : annos) {

            System.out.println(anno);
        }

    }

    /*
    	获取运行时类所实现的接口
     */
    @Test
    public void test5(){
        Class clazz = Person.class;
        Class[] interfaces = clazz.getInterfaces();
        for (Class anInterface : interfaces) {

            System.out.println(anInterface);
        }

    }
    /*
    	获取运行时类的带泛型的父类
     */
    @Test
    public void test6(){
        Class clazz = Person.class;
        Type genericSuperclass = clazz.getGenericSuperclass();
        System.out.println(genericSuperclass);//com.atguigu.java1.Creature<java.lang.String>
    }
}

🌔 5、获取泛型父类信息

/* Type:
 * (1)Class
 * (2)ParameterizedType   
 * 		例如:Father<String,Integer>
 * 			ArrayList<String>
 * (3)TypeVariable
 * 		例如:T,U,E,K,V
 * (4)WildcardType
 * 		例如:
 * 		ArrayList<?>
 * 		ArrayList<? super 下限>
 * 		ArrayList<? extends 上限>
 * (5)GenericArrayType
 * 		例如:T[]
 * 	
 */
public class TestGeneric {
	public static void main(String[] args) {
		//需求:在运行时,获取Son类型的泛型父类的泛型实参<String,Integer>
		
		//(1)还是先获取Class对象
		Class clazz = Son.class;//四种形式任意一种都可以
		
		//(2)获取泛型父类
//		Class sc = clazz.getSuperclass();
//		System.out.println(sc);
		/*
		 * getSuperclass()只能得到父类名,无法得到父类的泛型实参列表
		 */
		Type type = clazz.getGenericSuperclass();
		
		// Father<String,Integer>属于ParameterizedType
		ParameterizedType pt = (ParameterizedType) type;
		
		//(3)获取泛型父类的泛型实参列表
		Type[] typeArray = pt.getActualTypeArguments();
		for (Type type2 : typeArray) {
			System.out.println(type2);
		}
	}
}
//泛型形参:<T,U>
class Father<T,U>{
	
}
//泛型实参:<String,Integer>
class Son extends Father<String,Integer>{
	
}

🌔 6、获取内部类或外部类信息

  • public Class<?>[] getClasses(): 返回所有公共内部类和内部接口【包括从超类继承的公共类和接口成员以及该声明的公共类和接口成员】
  • public Class<?>[] getDeclaredClasses():返回Class对象的一个数组,这些对象梵音声明为此Class对象所表示的类的成员的所有类和接口与、包括该类所声明的公共、保护、默认(包)访问及私有类和接口,但不包括继承的类和接口
  • public Class<?> getDeclaringClass():如果此 Class 对象所表示的类或接口是一个内部类或内部接口,则返回它的外部类或外部接口,否则返回null
  • Class<?> getEnclosingClass() :返回某个内部类的外部类
	@Test
	public void test5(){
		Class<?> clazz = Map.class;
		Class<?>[] inners = clazz.getDeclaredClasses();
		for (Class<?> inner : inners) {
			System.out.println(inner);
		}
		
		Class<?> ec = Map.Entry.class;
		Class<?> outer = ec.getDeclaringClass();
		System.out.println(outer);
	}

4.3 调用运行时类的指定结构

🌔 1、调用指定的属性:

  • 在反射机制中,可以直接通过Field类操作类中的属性
    • Field 类提供的set()和get()方法就可以完成设置和获取属性内容

(1)获取该类型的Class对象

Class clazz = Class.forName(“包.类名”);

(2)获取属性对象

Field field = clazz.getDeclaredField(“属性名”);

(3) 如果属性的访问权限不是public,需要设置属性可访问

field.setAccessible(true);

(4)创建实例对象,如果操作的是非静态属性,需要创建实例对象
//如果有公共无参构造方法
Object obj = clazz.newInstance();
//如果想通过特定构造器对象创建实例对象
Object obj = 构造器对象.newInstance(实参…);

(5)设置指定对象obj上此Field的属性内容【就是给指定的对象设置field对应的属性值】

//如果是静态变量,可以将obj替换为null
field.set(obj, “属性值”);

(6)取得指定对象obj上此Field的属性内容
//如果操作静态变量,那么实例对象可以省略,用null表示
Object value = field.get(obj);

接下来用代码演示如何获取类中我们需要的指定结构

package com.zwh.shangguigu.reflection_;

import java.lang.reflect.Field;

/**
 * @author Bonbons
 * @version 1.0
 */
public class FieldTest {
    public static void main(String[] args) throws Exception{
        //1、先获取到Class类
        Class clazz = Class.forName("com.zwh.shangguigu.reflection_.Student");
        //2、获取属性对象
        Field idField = clazz.getDeclaredField("id");
        //3、如果id是私有属性,并且在当前类中没有访问权限
        idField.setAccessible(true);
        //4、创建实例对象
        Object stu = clazz.newInstance();
        //5、获取属性值
        /*
        * (1)正常情况下我们通过Student的对象获取属性值,调用getXxx()方法即可
        * (2)现在我们需要创建对应属性的对象,然后属性对象调用get()方法,将在内存中创建的Student对象作为参数传入
        * */
        Object value = idField.get(stu);
        System.out.println("id = " + value);

        //6、设置属性值
        /*
        * 以前:学生对象.setId(属性值);
        * 现在:id属性对象.set(学生对象,值);
        * */

        idField.set(stu, 33);
        value = idField.get(stu);
        System.out.println("id = " + value);
    }

}

//创建一个Student类
class Student{
    private String name;
    private int id;

    public Student(){}
    public Student(String name, int id) {
        this.name = name;
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", id=" + id +
                '}';
    }
}

在这里插入图片描述

🌔 2、补充说明关于 setAccessible方法的使用

  • Method、Field、Constructor对象都提供了这个方法
  • setAccessible 是启用和禁用访问安全检查的开关
  • 参数值为 true 则指示反射的对象在使用时应该取消Java语言访问检查
    • 可以提高反射的效率 【特别是需要大量用反射的情况下】
    • 可以使得本来无法访问的私有成员也可以访问
  • 参数值为 false则指示反射的对象应该实施Java语言的访问检查

🌔 3、调用指定的方法

请添加图片描述
(1)获取该类型的Class对象

Class clazz = Class.forName("包.类名");

(2)获取方法对象

Method method = clazz.getDeclaredMethod("方法名", 方法的形参类型列表);

(3)创建实例对象

Object obj = class.newInstance();

(4)调用方法

Object result = method.invoke(obj, 方法的实参值列表);

同获取指定的属性值一样:

如果方法的权限修饰符修饰的范围不可见,可以调用setAccessible(true);跳过检查
如果方法是静态方法,实例对象可以省略,用 null 代替

package com.zwh.shangguigu.reflection_;

import java.lang.reflect.Method;

/**
 * @author Bonbons
 * @version 1.0
 */
public class MethodTest {
    public static void main(String[] args) throws Exception{
        //1、获取Student的Class对象
        Class clazz = Class.forName("com.zwh.shangguigu.reflection_.Student");
        //2、创建我们需要的方法的对象
        Method setNameMethod = clazz.getDeclaredMethod("setName", String.class);
        //3、创建Student的实例
        Object stu = clazz.newInstance();
        //4、调用方法
        Object setNameMethodReturnValue = setNameMethod.invoke(stu, "张三");
        System.out.println("stu = " + stu);
        //set方法都是void,所以下面输入的肯定是null
        System.out.println("setNameMethodReturnValue = " + setNameMethodReturnValue);

        //5、为了演示调用静态方法,我在Student类中添加一个
        //接下来需要获取方法的对象,至于Class的对象就用上面的那个了
        Method printMethod = clazz.getDeclaredMethod("print", String.class);
        //调用方法,因为是静态方法,所以不用传入Student的实例
        printMethod.invoke(null, "我想打印一张照片");

    }
}

在这里插入图片描述
🌔 4、案例分析

  • 读取user.properties文件中的数据,通过反射完成User类对象的创建及对应方法的调用

在这里插入图片描述

我们需要准备对应的实体类

package com.atguigu.bean;

public class User {
    private String name;

    public User() {
    }

    public User(String name) {
        this.name = name;
    }

    public void show(){
        System.out.println("我是一个焦绿平台的用户");
    }
}

编写我们的反射文件 ReflectTest

package com.atguigu.java4;

import org.junit.Test;

import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Method;
import java.util.Properties;

public Class ReflectTest{
	public static void main(String [] args){
		//1、创建Properties对象
		Properties pro = new Properties();

		//2、加载配置文件,转换成一个集合(就是通过类加载器将数据转换成流)
		ClassLoader classLoader = ClassLoader.getSystemClassLoader();
		InputStream is = classLoader.getResourceAsStream("user.properties");
		pro.add(is);

		//3、获取配置文件中的数据
		String className = pro,getProperty("className");
		String methodName = pro.getProperty("methodName");
		
		//4、将这个类加载到内存中
		Class clazz = Class.forName(className);

		//5、创建对象
		Object instance = clazz.newInstance();

		//6、获取方法对象
		Method showMethod = clazz.getMethod(methodName);

		//7、执行方法
		showMethod.invoke(iinstance);		
	}
}

五、读取注解信息

  • 一个完整的注解应该包含三部分:声明、使用、读取

🌔 1、声明自定义注解

package com.zwh.shangguigu.reflection_;

import java.lang.annotation.*;

@Inherited
@Target(ElementType.TYPE)
@Rentention(RententionPolicy.RUNTIME)
public @interface Table{
	String value();
}
package com.atguigu.annotation;

import java.lang.annotation.*;

@Inherited
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Column {
    String columnName();
    String columnType();
}
  • 自定义注解可以通过四个元注解分别说明以下情况:

    • @Retention 声明周期
    • @Target 使用位置
    • @Inherited 是否被继承
    • @Documented 是否被生成API文档中
  • Annotation 的成员在 Annotation 定义中以无参数有返回值的抽象方法的形式来声明,我们又称为配置参数。返回值类型只能是八种基本数据类型、String类型、Class类型、enum类型、Annotation类型、以上所有类型的数组

    • 可以使用 default 关键字为抽象方法指定默认返回值如果定义的注解含有抽象方法,那么使用时必须指定返回值,除非它有默认值。
    • 格式是“方法名 = 返回值”,如果只有一个抽象方法需要赋值,且方法名为value,可以省略“value=”,所以如果注解只有一个抽象方法成员,建议使用方法名value

🌔 2、使用自定义注解

package com.atguigu.annotation;

@Table("t_stu")
public class Student {
    @Column(columnName = "sid",columnType = "int")
    private int id;
    @Column(columnName = "sname",columnType = "varchar(20)")
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Student{" +
                "id=" + id +
                ", name='" + name + '\'' +
                '}';
    }
}

🌔 3、读取和处理自定义注解

  • 自定义注解必须配上注解的信息处理流程才有意义
  • 我们自定义注解,只能通过反射代码读取,所以自定义注解的声明周期必须是:RetentionPolicy.RUNTIME
package com.atguigu.annotation;

import java.lang.reflect.Field;

public class TestAnnotation {
    public static void main(String[] args) {
        Class studentClass = Student.class;
        Table tableAnnotation = (Table) studentClass.getAnnotation(Table.class);
        String tableName = "";
        if(tableAnnotation != null){
            tableName = tableAnnotation.value();
        }

        Field[] declaredFields = studentClass.getDeclaredFields();
        String[] columns = new String[declaredFields.length];
        int index = 0;
        for (Field declaredField : declaredFields) {
            Column column = declaredField.getAnnotation(Column.class);
            if(column!= null) {
                columns[index++] = column.columnName();
            }
        }
        
        String sql = "select ";
        for (int i=0; i<index; i++) {
            sql += columns[i];
            if(i<index-1){
                sql += ",";
            }
        }
        sql += " from " + tableName;
        System.out.println("sql = " + sql);
    }
}

六、体会反射的动态性

🌔 1、案例一

public class ReflectionTest {

    //体会反射的动态性:动态的创建给定字符串对应的类的对象
    public <T> T getInstance(String className) throws Exception {

        Class clazz = Class.forName(className);

        Constructor constructor = clazz.getDeclaredConstructor();
        constructor.setAccessible(true);
        return (T) constructor.newInstance();

    }

    @Test
    public void test1() throws Exception {
        String className = "com.atguigu.java1.Person";
        Person p1 = getInstance(className);
        System.out.println(p1);
    }
}

🌔 2、案例二

public class ReflectionTest {
    //体会反射的动态性:动态的创建指定字符串对应类的对象,并调用指定的方法
    public Object  invoke(String className,String methodName) throws Exception {
        Class clazz = Class.forName(className);
        Constructor constructor = clazz.getDeclaredConstructor();
        constructor.setAccessible(true);
        //动态的创建指定字符串对应类的对象
        Object obj = constructor.newInstance();

        Method method = clazz.getDeclaredMethod(methodName);
        method.setAccessible(true);
        return method.invoke(obj);
    }

    @Test
    public void test2() throws Exception {
        String info = (String) invoke("com.atguigu.java1.Person", "show");

        System.out.println("返回值为:" + info);

    }
}

🌔 3、案例三

public class ReflectionTest {
	@Test
    public void test1() throws Exception {
        //1.加载配置文件,并获取指定的fruitName值
        Properties pros = new Properties();
        InputStream is = ClassLoader.getSystemClassLoader().getResourceAsStream("config.properties");
        pros.load(is);
        String fruitStr = pros.getProperty("fruitName");
        //2.创建指定全类名对应类的实例
        Class clazz = Class.forName(fruitStr);
        Constructor constructor = clazz.getDeclaredConstructor();
        constructor.setAccessible(true);
        Fruit fruit = (Fruit) constructor.newInstance();
        //3. 调用相关方法,进行测试
        Juicer juicer = new Juicer();
        juicer.run(fruit);

    }

}

interface Fruit {
	public void squeeze();
}

class Apple implements Fruit {
	public void squeeze() {
		System.out.println("榨出一杯苹果汁儿");
	}
}

class Orange implements Fruit {
	public void squeeze() {
		System.out.println("榨出一杯桔子汁儿");
	}
}

class Juicer {
	public void run(Fruit f) {
		f.squeeze();
	}
}

其中,配置文件【config.properties】存放在当前Module的src下

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/2052.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

1、Linux初级——linux命令

下载镜像&#xff1a;http://cn.ubuntu.com/dowload 一、基本命令 1、alias&#xff08;给命令取别名&#xff09; 例如&#xff1a;alias clls -la&#xff08;只是临时的&#xff09; 2、配置文件$ vim ~/.bashrc $ vim ~/.bashrc // 使用vim打开配置文件 (1)在配置文件…

初识STM32单片机

目录 一、单片机基本认知 二、STM系列单片机命名规则 三、标准库与HAL库区别 四、通用输入输出端口GPIO 五、推挽输出与开漏输出 六、复位和时钟控制&#xff08;RCC&#xff09; 七、时钟控制 八、中断和事件 九、定时器介绍 一、单片机基本认知 单片机和PC电脑相比…

搜索系统(二)

term weight 如何衡量一个词在一篇文档中的重要性 词频率&#xff08;tf&#xff09;&#xff1a;term在文档中出现了多少次&#xff0c;tf越大说明越重要 逆文档频率&#xff08;idf&#xff09;&#xff1a;有多少文档包含了这个term&#xff0c;idf越大表明越不重要 综合…

Unity --- 游戏案例 --- 英雄无敌与Ruby

1.如何在场景中标识出一个空游戏物体&#xff08;对象集群&#xff09; 1.选中该空游戏物体&#xff0c;然后在Inspector面板处的物体名旁边添加想要的颜色的图标即可&#xff0c;最终效果如下图 2.这个图标只在场景中能开到&#xff0c;在游戏画面中是看不到的&#xff0c;其存…

Vulnhub项目:Web Machine(N7)

靶机地址&#xff1a;Web Machine(N7)渗透过程&#xff1a;kali ip&#xff1a;192.168.56.104&#xff0c;靶机ip&#xff0c;使用arp-scan进行查看靶机地址&#xff1a;192.168.56.128收集靶机开放端口&#xff1a;nmap -sS -sV -T5 -A 192.168.56.128开放了80端口&#xff0…

索尼mxf变成rsv的修复方法

索尼的影视级摄像机一般是用MXF文件结构&#xff0c;在一些极端情况下(如断电)会生成RSV文件&#xff0c;遇到这种情况我们应该如何处理&#xff1f;下面来看看今天这个案例。故障文件:12.51G RSV文件故障现象:断电后仅生成了一个扩展名为rsv的文件&#xff0c;使用播放器可以播…

python+django+vue全家桶鲜花商城售卖系统

重点&#xff1a; &#xff08;1&#xff09; 网上花店网站中各模块功能之间的的串联。 &#xff08;2&#xff09; 网上花店网站前台与后台的连接与同步。 &#xff08;3&#xff09; 鲜花信息管理模块中鲜花的发布、更新与删除。 &#xff08;4&#xff09; 订单…

java多线程之线程的六种状态

线程的六种状态(1) NEW(初始状态)(2) TERMINATED(终止状态 / 死亡状态)(3) RUNNABLE(运行时状态)(4) TIMED_WAITING(超时等待状态)(5) WAITING(等待状态)(6) BLOCK(阻塞状态)sleep和wait的区别:操作系统里的线程自身是有一个状态的,但是java Thread 是对系统线程的封装,把这里的…

基于C++的AI五子棋游戏项目开发教程

项目资源下载 基于C的AI五子棋游戏项目源码压缩包下载地址基于C的AI五子棋游戏项目源码Github下载地址基于C的AI五子棋游戏项目所需素材基于C的AI五子棋游戏项目所需要的EasyX 项目简介 本项目基于C开发&#xff0c;整体来说比较简单&#xff0c;实现了人与AI之间的五子棋对弈…

Java实习生------Redis常见面试题汇总(AOF持久化、RDB快照、分布式锁、缓存一致性)⭐⭐⭐

“年轻人&#xff0c;就要勇敢追梦”&#x1f339; 参考资料&#xff1a;图解redis 目录 谈谈你对AOF持久化的理解&#xff1f; redis的三种写回策略是什么&#xff1f; 谈谈你对AOF重写机制的理解&#xff1f;AOF重写机制的具体过程&#xff1f; 谈谈你对RDB快照的理解&a…

面试官:说一下MySQL中的锁机制吧

5. 1MySQL有哪些锁&#xff1f; 为保证数据的一致性&#xff0c;需要对并发操作进行控制&#xff0c;因此产生了锁。同时锁机制也为实现MySQL的各个隔离级别提供了保证。 锁冲突 也是影响数据库并发访问性能的一个重要因素。所以锁对数据库而言显得尤其重要&#xff0c;也更加…

seata服务搭建

它支持两种存储模式&#xff0c;一个是文件&#xff0c;一个是数据库&#xff0c;下面我们分别介绍一下这两种配置nacos存储配置&#xff0c;注意如果registry.conf中注册和配置使用的是file&#xff0c;就会去读取file.config的配置&#xff0c;如果是nacos则通过nacos动态读取…

Kafka和RabbitMQ有哪些区别,各自适合什么场景?

目录标题1. 消息的顺序2. 消息的匹配3. 消息的超时4. 消息的保持5. 消息的错误处理6. 消息的吞吐量总结1. 消息的顺序 有这样一个需求&#xff1a;当订单状态变化的时候&#xff0c;把订单状态变化的消息发送给所有关心订单变化的系统。 订单会有创建成功、待付款、已支付、已…

Cookie和Session详解

目录 前言&#xff1a; Session详解 Cookie和Session区别和关联 服务器组织会话的方式 使用Tomcat实现登录成功跳转到欢迎页面 登录前端页面 登录成功后端服务器 重定向到欢迎页面 抓包分析交互过程 小结&#xff1a; 前言&#xff1a; Cookie之前博客有介绍过&#x…

音视频技术开发周刊 | 285

每周一期&#xff0c;纵览音视频技术领域的干货。新闻投稿&#xff1a;contributelivevideostack.com。GPT-4 Office全家桶发布谷歌前脚刚宣布AI工具整合进Workspace&#xff0c;微软后脚就急匆匆召开了发布会&#xff0c;人狠话不多地祭出了办公软件王炸——Microsoft 365 Cop…

使用GPT-4生成QT代码

一、概述最近ChatGPT火爆起来了&#xff0c;ChatGPT是一种基于GPT的自然语言处理模型&#xff0c;可以用于生成自然语言文本&#xff0c;例如对话、文章等。最近又发现了一个优秀且免费的代码生成工具Cursor.so &#xff0c;Cursor.so集成了 GPT-4 &#xff0c;可以帮助你快速编…

Python3,5行代码,生成自动排序动图,这操作不比Excel香?

5行代码生成自动排序动图1、引言2、代码实战2.1 pynimate介绍2.2 pynimate安装2.3 代码示例3、总结1、引言 小屌丝&#xff1a;鱼哥&#xff0c;听说你的excel段位又提升了&#xff1f; 小鱼&#xff1a;你这是疑问的语气&#xff1f; 小屌丝&#xff1a;没有~ 吧… 小鱼&…

计算机网络复习重点

文章目录计算机网络复习重点第一章 计算机网络和因特网概念与应用1、什么是因特网2、协议protocol3、入网方式4、物理媒介5、数据交换模式6、延时与丢包什么时候发生延时&#xff1f;延时的类型丢包何时发生7、协议层次与模型因特网协议栈TCP / IP模型ISO/OSI参考模型协议数据单…

MySQL-存储过程

什么是存储过程我们前面所学习的MySQL语句都是针对一个表或几个表的单条 SQL 语句&#xff0c;但是在数据库的实际操作中&#xff0c;并非所有操作都那么简单&#xff0c;经常会有一个完整的操作需要多条SQL语句处理多个表才能完成。例如&#xff0c;为了确认学生能否毕业&…

归并排序和快速排序

目录 归并排序 思路&#xff1a; 代码执行&#xff1a; 快速排序 运行流程图&#xff1a; 代码思路&#xff1a; 代码执行&#xff1a; 归并排序 定义&#xff1a;归并排序是建立在归并操作上的一种有效&#xff0c;稳定的排序算法&#xff0c;该算法是采用分治法&#x…