十九、反射

一、反射机制

1.1 反射机制的快速入门

在这里插入图片描述

package com.gyh.reflection.question;

import com.gyh.Cat;

import java.io.FileInputStream;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Properties;

/**
 * @author Gao YongHao
 * @version 1.0
 */
public class ReflectionQuestion {
    public static void main(String[] args) throws IOException, ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        // 传统的方式 new 对象 --> 调用方法
//        Cat cat = new Cat();
//        cat.hi();

        // 我们尝试做一做 ---> 明白反射

        // 1. 使用Properties类,可以读写配置文件
        Properties properties = new Properties();
        properties.load(new FileInputStream("src\\re.properties"));
        String classfullpath = properties.getProperty("classfullpath");
        String methodName = properties.getProperty("method");
        System.out.println("classfullpath" + classfullpath);
        System.out.println("method" + methodName);

        // 2. 创建对象,  传统的方法,  行不通 ===> 反射机制

        // 3. 使用反射机制解决
        // (1) 加载类,返回Class类型的对象cls
        Class<?> cls = Class.forName(classfullpath);
        // (2) 通过 cls 得到你加载的类 com.gyh.Cat 的对象实例
        Object o = cls.newInstance();
        System.out.println(o instanceof Cat);// 运行类型
        // (3) 通过 cls 得到你加载的类 com.gyh.Cat 的 methodName"hi" 的方法对象
        //  即:在反射中,可以把方法视为对象(万物皆对象)
        Method method1 = cls.getMethod(methodName);
        // (4) 通过 method1 调用方法:即通过方法对象来实现调用方法
        method1.invoke(o); // 传统方法:对象.方法()  ,反射机制:方法.invoke(对象)

    }
}

1.2 反射机制的介绍

在这里插入图片描述

在这里插入图片描述

1.3 反射机制的主要类

在这里插入图片描述

在这里插入图片描述

package com.gyh.reflection.question;

import com.gyh.Cat;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Properties;

/**
 * @author Gao YongHao
 * @version 1.0
 */
public class Reflection01 {
    public static void main(String[] args) throws IOException, ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        // 1. 使用Properties类,可以读写配置文件
        Properties properties = new Properties();
        properties.load(new FileInputStream("src\\re.properties"));
        String classfullpath = properties.getProperty("classfullpath");
        String methodName = properties.getProperty("method");

        // (1) 加载类,返回Class类型的对象cls
        Class<?> cls = Class.forName(classfullpath);
        // (2) 通过 cls 得到你加载的类 com.gyh.Cat 的对象实例
        Object o = cls.newInstance();
        System.out.println(o instanceof Cat);// 运行类型
        // (3) 通过 cls 得到你加载的类 com.gyh.Cat 的 methodName"hi" 的方法对象
        //  即:在反射中,可以把方法视为对象(万物皆对象)
        Method method1 = cls.getMethod(methodName);
        // (4) 通过 method1 调用方法:即通过方法对象来实现调用方法
        method1.invoke(o); // 传统方法:对象.方法()  ,反射机制:方法.invoke(对象)

        // 得到name字段
        // getFiled不能得到私有的属性!
        Field age = cls.getField("age");
        System.out.println(age.get(o)); // 传统写法 对象.成员变量  , 反射:成员变量对象.get(对象)

        // () 中可以指定构造器参数类型, 返回的是无参构造器
        Constructor<?> constructor = cls.getConstructor();
        System.out.println(constructor);

        // 传入的 String.class 就是String 类的 Class 对象
        Constructor<?> constructor1 = cls.getConstructor(String.class, Integer.class);
        System.out.println(constructor1);
    }
}

1.4 反射优点和缺点

在这里插入图片描述

在这里插入图片描述

package com.gyh.reflection.question;

import com.gyh.Cat;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
 * @author Gao YongHao
 * @version 1.0
 * <p>
 * 测试反射调用的性能
 */
public class Reflection02 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {
        m1();
        m2();
        m3();
    }

    // 传统方法来调用hi
    public static void m1() {
        Cat cat = new Cat();
        long start = System.currentTimeMillis();
        for (int i = 0; i < 90000000; i++) {
            cat.hi();
        }
        long end = System.currentTimeMillis();
        System.out.println("传统方法来调用hi 耗时=" + (end - start));
    }

    // 反射机制调用方法1
    public static void m2() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        Class<?> cls = Class.forName("com.gyh.Cat");
        Object o = cls.newInstance();
        Method method = cls.getMethod("hi");
        long start = System.currentTimeMillis();
        for (int i = 0; i < 90000000; i++) {
            method.invoke(o);
        }
        long end = System.currentTimeMillis();
        System.out.println("基于反射的方法来调用hi 耗时=" + (end - start));
    }

    // 反射机制调用方法2
    public static void m3() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        Class<?> cls = Class.forName("com.gyh.Cat");
        Object o = cls.newInstance();
        Method method = cls.getMethod("hi");
        method.setAccessible(true); // 在反射调用方法时,取消访问检测
        long start = System.currentTimeMillis();
        for (int i = 0; i < 90000000; i++) {
            method.invoke(o);
        }
        long end = System.currentTimeMillis();
        System.out.println("基于反射(取消访问检测)的方法来调用hi 耗时=" + (end - start));
    }
}

二、Class类

2.1 基本介绍

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

2.2 Class类常用方法

在这里插入图片描述

package com.gyh.class_;

import java.lang.reflect.Field;

/**
 * @author Gao YongHao
 * @version 1.0
 * 演示Class类的常用方法
 */
public class Class02 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchFieldException {
        String classAllPath = "com.gyh.Car";
        // 1. 获取到Car类 对应的 Class 对象
        // <?> 表示不确定的Java类型
        Class<?> cls = Class.forName(classAllPath);
        // 2. 输出 cls
        System.out.println(cls); // 显示 cls 对象,是哪个类的Class对象 com.gyh.Car
        System.out.println(cls.getClass()); // 输出运行类型 java.lang.Class
        // 3. 得到包名
        System.out.println(cls.getPackage().getName()); // 包名
        // 4. 得到全类名
        System.out.println(cls.getName());
        // 5. 通过cls来创建对象实例
        Object o = cls.newInstance();
        System.out.println(o);
        // 6. 通过反射获取属性 brand
        Field brand = cls.getField("brand");
        System.out.println(brand.get(o));
        // 7. 通过反射给属性赋值
        brand.set(o, "奔驰");
        System.out.println(brand.get(o));
        // 8. 希望得到所有的属性
        Field[] fields = cls.getFields();
        for (Field f : fields) {
            System.out.println(f.getName());
        }
    }
}

2.3 获取Class类对象

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

package com.gyh.class_;

import com.gyh.Car;

/**
 * @author Gao YongHao
 * @version 1.0
 * 演示得到Class对象的各种方式(6)
 */
public class GetClass_ {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
        // 1. Class.fromName
        String classAllPath = "com.gyh.Car";
        Class<?> cls1 = Class.forName(classAllPath);
        System.out.println(cls1);

        // 2. 类名.class
        Class<Car> cls2 = Car.class;
        System.out.println(cls2);

        // 3. 通过对象.getClass() 应用场景,有对象实例
        Car car = new Car();
        Class<? extends Car> cls3 = car.getClass();
        System.out.println(cls3);

        // 4. 通过类加载器[4种]来获取到类的Class对象
        // (1) 先得到类加载器 car
        ClassLoader classLoader = car.getClass().getClassLoader();
        // (2) 通过类加载器得到Class对象
        Class<?> cls4 = classLoader.loadClass(classAllPath);
        System.out.println(cls4);

        // cls1, cls2, cls3, cls4 是同一个对象
        System.out.println(cls1.hashCode());
        System.out.println(cls2.hashCode());
        System.out.println(cls3.hashCode());
        System.out.println(cls4.hashCode());

        // 5. 基本数据(byte, short ,int, long,char, boolean, float, double)
        // 按如下方式得到Class类对象
        Class<Integer> integerClass = int.class;
        Class<Character> characterClass = char.class;
        Class<Boolean> booleanClass = boolean.class;
        System.out.println(integerClass);

        // 6. 基本数据类型对应的包装类,可以通过 .TYPE 的得到Class类对象
        Class<Integer> type = Integer.TYPE;
        Class<Character> type1 = Character.TYPE;
        System.out.println(type);
        System.out.println(type1);

        System.out.println(integerClass.hashCode());
        System.out.println(type.hashCode());
    }
}

2.4 哪些类型有Class对象

在这里插入图片描述

package com.gyh.class_;

import java.io.Serializable;

/**
 * @author Gao YongHao
 * @version 1.0
 * 演示哪些类型有Class对象
 */
public class AllTypeClass {
    public static void main(String[] args) {
        Class<String> stringClass = String.class;// 外部类
        Class<Serializable> serializableClass = Serializable.class;// 接口
        Class<Integer[]> aClass = Integer[].class; // 数组
        Class<Integer[][]> aClass1 = Integer[][].class; // 二维数组
        Class<Deprecated> deprecatedClass = Deprecated.class; // 注解
        // 枚举
        Class<Thread.State> stateClass = Thread.State.class; // 枚举类
        Class<Long> longClass = long.class; // 基本数据类型
        Class<Void> voidClass = void.class;
        Class<Class> classClass = Class.class;
    }
}

三、类加载

3.1 静态加载和动态加载

  • JAVA动态加载类静态加载类的区别 使用new等方式创建对象的方式称作为静态加载,而使用Class.forName(“XXX”)称作为动态加载,它们俩本质的区别在于静态加载的类的源程序在编译时期加载(必须存在),而动态加载的类在编译时期可以缺席(源程序不必存在)。

  • 为什么需要动态加载类 对于我自己的理解,动态加载类增加了程序的灵活性。比如一个程序中有50个功能,但你可能只会使用其中的一个,如果你用的是静态加载的方式,你必须在编译前提供100个功能的所有定义,否则无法编译通过,若你使用的是动态加载机制,则不需要如此大费周章,用哪一个就定义哪一个即可。

摘自:https://www.zhihu.com/question/503266400/answer/2254504600

在这里插入图片描述

package com.gyh.class_;

// 因为 new Car() 是静态加载因此必须编写(导入)Car类
import com.gyh.Car;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Scanner;

/**
 * @author Gao YongHao
 * @version 1.0
 */
public class ClassLoad_ {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入key:");
        String key = scanner.next();
        switch (key) {
            case "1":
                Car car = new Car(); // 静态加载,依赖性很强。在编译时需要该类的源文件存在
                car.drive();
                //....
                break;
            case "2":
                // 反射 --> 动态加载
                Class<?> cls = Class.forName("Person");// 加载 Person 类【动态加载】,在编译时不需要该类的源文件存在,只在运行时检测
                Object o = cls.newInstance();
                Method drive = cls.getMethod("drive");
                drive.invoke(o);
                break;
            default:
                System.out.println("do nothing...");
        }

    }
}

3.2 类加载过程图

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

class ClasLoad02{
    public static void main(String[] args){
        
    }
}

class A{
    // 属性 - 成员变量 - 字段
    // 分析类加载的 链接阶段-准备 属性是如何处理
    // 1. n1 是实例属性,不是静态变量,因此在准备阶段,是不会分配内存
    // 2. n2 是静态变量,分配内存 n2 是默认初始化 0,而不是 20
    // 3. n3 是 static final 常量,它和静态变量不一样,因为一旦赋值就不变 n3  = 30
    public int n1 = 10;
    public static int n2 = 20;
    public static final int n3 = 30;
}

在这里插入图片描述

在这里插入图片描述

package com.gyh.class_;

/**
 * @author Gao YongHao
 * @version 1.0
 */
public class classLoader03 {
    public static void main(String[] args) {
        // 分析
        // 1. 加载B类,并生成 B 的class对象
        // 2. 连接 num = 0 (准备操作:分配内存并默认初始化)
        // 3. 初始化阶段
        //    依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并合并
        /**
         * clinit(){
         *     System.out.println("B 静态代码块被执行");
         *     // num =300;
         *     num = 100;
         * }
         * 合并:num = 100
         *
         */
        
//        new B(); // 类加载
        System.out.println(B.num); // 100

    }
}

class B {
    static {
        System.out.println("B 静态代码块被执行");
        num = 300;
    }

    static int num = 100;

    public B() {
        System.out.println("B() 构造器被执行");
    }
}

四、反射获取类的结构信息

在这里插入图片描述

package com.gyh.class_;

import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

/**
 * @author Gao YongHao
 * @version 1.0
 * 演示如何通过反射获取类的结构信息
 */
public class ReflectionUtils {
    public static void main(String[] args) throws Exception {
        api_01();
    }

    public static void api_01() throws Exception {
        // 得到Class对象
        Class<?> personClass = Class.forName("com.gyh.class_.Person");

        // getName: 获取全类名
        System.out.println(personClass.getName());

        // getSimpleName: 获取简单类名
        System.out.println(personClass.getSimpleName());

        // getField:获取所有public修饰的属性,包含本类以及父类
        Field[] fields = personClass.getFields();
        for (Field field : fields) {
            System.out.println("本类以及父类的属性" + field.getName());
        }

        // getDeclaredFields: 获取本类中所有属性
        Field[] declaredFields = personClass.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            System.out.println("本类的属性" + declaredField.getName());
        }

        // getMethods:获取所有public修饰的方法,包含本类以及父类
        Method[] methods = personClass.getMethods();
        for (Method method : methods) {
            System.out.println("本类以及父类的方法" + method.getName());
        }

        // getDeclaredMethods:获取本类中所有方法
        Method[] declaredMethods = personClass.getDeclaredMethods();
        for (Method declaredMethod : declaredMethods) {
            System.out.println("本类的方法" + declaredMethod.getName());
        }

        // getConstructors:获取所有 public 修饰的构造器,包含本类以及父类
        Constructor<?>[] constructors = personClass.getConstructors();
        for (Constructor<?> constructor : constructors) {
            System.out.println("父类以及本类的构造器" + constructor.getName());
        }

        // getDeclaredConstructors:获取本类所有的构造器
        Constructor<?>[] declaredConstructors = personClass.getDeclaredConstructors();
        for (Constructor<?> c : declaredConstructors) {
            System.out.println("本类的构造器" + c.getName());
        }

        // getPackage:以Package形式返回 包信息
        System.out.println(personClass.getPackage());

        // getSuperclass:以Class形式返回父类信息
        System.out.println(personClass.getSuperclass());

        // getInterfaces:以Class[]形式返回接口信息
        Class<?>[] interfaces = personClass.getInterfaces();
        for (Class<?> anInterface : interfaces) {
            System.out.println(anInterface.getName());
        }

        // getAnnotations: 以 Annotation[] 形式返回注解信息
        Annotation[] annotations = personClass.getAnnotations();
        for (Annotation annotation : annotations) {
            System.out.println(annotation);
        }
    }
}

class Human {
    public String key;

    public Human() {
    }

    public void hi() {
        System.out.println("hi");
    }
}

interface BI {

}

interface AI {

}

@Deprecated()
class Person extends Human implements AI, BI {
    // 属性
    public String name;
    protected int age;
    String job;
    private double salary;

    public Person() {
    }

    // 方法
    public void m1() {

    }

    protected void m2() {

    }

    void m3() {

    }

    private void m4() {

    }
}

在这里插入图片描述

在这里插入图片描述

package com.gyh.class_;

import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

/**
 * @author Gao YongHao
 * @version 1.0
 * 演示如何通过反射获取类的结构信息
 */
public class ReflectionUtils {
    public static void main(String[] args) throws Exception {
        api_02();
    }

    public static void api_02() throws Exception {
        // 得到Class对象
        Class<?> personClass = Class.forName("com.gyh.class_.Person");
        Field[] declaredFields = personClass.getDeclaredFields();
        // 规定  说明:
        for (Field declaredField : declaredFields) {
            System.out.println("本类的属性" + declaredField.getName() +
                    " 该属性的修饰符值" + declaredField.getModifiers()
                    + " 该属性的类型" + declaredField.getType());
        }

        // getMethods:获取所有public修饰的方法,包含本类以及父类
        Method[] methods = personClass.getMethods();
        for (Method method : methods) {
            System.out.print("本类以及父类的方法" + method.getName() +
                    " 该方法的访问修饰符" + method.getModifiers() +
                    " 该方法返回类型" + method.getReturnType());
            Class<?>[] parameterTypes = method.getParameterTypes();
            for (Class<?> parameterType : parameterTypes) {
                System.out.print("该方法的形参列表类型" + parameterType);
            }
            System.out.println();
        }

    }
}

class Human {
    public String key;

    public Human() {
    }

    public void hi() {
        System.out.println("hi");
    }
}

interface BI {

}

interface AI {

}

@Deprecated()
class Person extends Human implements AI, BI {
    // 属性
    public String name;
    protected int age;
    String job;
    private double salary;

    public Person() {
    }

    // 方法
    public void m1() {

    }

    protected void m2() {

    }

    void m3() {

    }

    private void m4() {

    }
}

五、通过反射创建对象

在这里插入图片描述

在这里插入图片描述

package com.gyh.class_;

import java.io.IOException;
import java.lang.reflect.Constructor;

/**
 * @author Gao YongHao
 * @version 1.0
 */
public class ReflectCreateInstance {
    public static void main(String[] args) throws Exception {
        // 1. 先获取到User类的Class对象
        Class<?> userClass = Class.forName("com.gyh.class_.User");

        // 2. 通过public的无参构造器创建实例
        Object o = userClass.newInstance();
        System.out.println(o);

        // 3. 通过public的有参构造器创建实例
        // 3.1 先得到对应的构造器
        Constructor<?> constructor = userClass.getConstructor(String.class);
        // 3.2 创建实例,并传入实参
        Object o1 = constructor.newInstance("韩顺平");
        System.out.println(o1);

        // 4. 通过非public的有参构造器创建实例
        Constructor<?> declaredConstructor = userClass.getDeclaredConstructor(int.class, String.class);
        declaredConstructor.setAccessible(true); // 爆破,使用反射可以访问private构造器
        Object gyh = declaredConstructor.newInstance(12, "gyh");
        System.out.println(gyh);
    }
}

class User { // User类
    private int age = 10;
    private String name = "gyh";

    public User() {

    }

    public User(String name) { // public 有参构造器
        this.name = name;
    }

    private User(int age, String name) { // private 有参构造器
        this.age = age;
        this.name = name;
    }

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

六、通过反射访问类中的成员

6.1 访问属性

在这里插入图片描述

package com.gyh.class_;

import java.lang.reflect.Field;

/**
 * @author Gao YongHao
 * @version 1.0
 */
public class ReflectAccessProperty {
    public static void main(String[] args) throws Exception {
        // 1. 得到Student类对应的 Class 对象
        Class<?> stu = Class.forName("com.gyh.class_.Student");
        // 2. 创建对象
        Object o = stu.newInstance(); // o的运行类型就是Student
        System.out.println(o.getClass()); // Student
        // 3. 使用反射得到age属性对象
        Field age = stu.getField("age");
        age.set(o, 88); // 通过反射来操作属性
        System.out.println(o); //

        // 4. 使用反射操作 name 属性
        Field name = stu.getDeclaredField("name");
        name.setAccessible(true);
        name.set(null, "hsp"); // 因为name是static属性,因此 o 也可以写出null
        System.out.println(o);
        System.out.println(name.get(o)); // 获取属性值
        System.out.println(name.get(null)); // 获取属性值,要求name是static

    }
}

class Student { // 类
    public int age;
    private static String name;

    public Student() { // 构造器

    }

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

6.2 访问方法

在这里插入图片描述

package com.gyh.class_;

import java.lang.reflect.Method;

/**
 * @author Gao YongHao
 * @version 1.0
 */
public class ReflectAssessMethod {
    public static void main(String[] args) throws Exception {
        // 1. 获取Boss类对应的Class对象
        Class<?> bossClass = Class.forName("com.gyh.class_.Boss");
        // 2. 创建对象
        Object o = bossClass.newInstance();
        // 3. 调用 public 的hi方法
        Method hi = bossClass.getMethod("hi", String.class);
        hi.invoke(o, "siri");
        // 4. 调用 private static 的 say 方法
        // 4.1 得到 say 方法对象
        Method say = bossClass.getDeclaredMethod("say", int.class, String.class, char.class);
        // 4.2 因为say方法是private,所以需要爆破,原理和前面讲的构造器和属性一样
        say.setAccessible(true); // 爆破
        // 在反射中,如果方法有返回值,统一返回Object
        Object invoke = say.invoke(o, 1, "abc", 'a');
        System.out.println(invoke);
        // 4.3 因为say方法是static的,还可以这样调用,可以传入null
        Object invoke1 = say.invoke(null, 1, "abc", 'a');
        System.out.println(invoke1);

    }
}

class Boss { // 类
    public int age;
    private static String name;

    public Boss() {
    }


    public void hi(String s) {
        System.out.println("hi " + s);
    }

    private static String say(int n, String s, char c) {
        return n + "" + s + "" + c;
    }
}


细节

  • 对于Class类中的 getXXXgetDeclaredXXX 方法,前者获取所有public修饰成员,后者获取所有成员(不限制修饰符
  • 15
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ModelBulider

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值