Java编程基础之反射

0.反射问题的引入

一个需求引出反射
1.根据配置文件re.properties指定信息,创建对象并调用方法

classfullpath=com.hspedu.Cat
method=hi

2.这样的需求在学习框架时特别多,即通过外部文件配置,在不修改源码情况下,来控制程序,也符合设计模式的ocp原则(开闭原则:不修改源码,扩展功能
3.快速入门com.hspedu.reflection.question
ReflectionQuestion.java

代码:
com.hspedu.Cat

package com.hspedu;

public class Cat {

    private String name = "招财猫";

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

    public void cry() {
        System.out.println("喵喵叫...");
    }
}

com.hspedu.reflection.question.ReflectionQuestion

package com.hspedu.reflection.question;

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

/**
 * 反射问题的引入
 */
@SuppressWarnings({"all"})
public class ReflectionQuestion {
    public static void main(String[] args) throws IOException, ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        //根据配置文件re.properties指定信息,创建对象并调用方法

        //传统方式:new 对象=>调用方法
//        Cat cat = new Cat();
//        cat.hi(); ==> cat.cry() 修改源码

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

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

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

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

反射最牛逼的地方是:不修改源码的情况下,扩展功能,即设计模式的ocp原则(开闭原则)

1.反射机制

1.1.Java Reflection

  1. 反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息(比如成员变量,构造器,成员方法等等),并能操作对象的属性及方法。反射在设计模式和框架底层都会用到
  2. 加载完类之后,在堆中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象包含了类的完整结构信息。通过这个对象得到类的结构。这个Class对象就像一面镜子,透过这个镜子看到类的结构,所以,形象的称之为:反射

1.2.Java反射机制原理示意图

1.3.Java反射机制可以完成

  1. 在运行时判断任意一个对象所属的类
  2. 在运行时构造任意一个类的对象
  3. 在运行时得到任意一个类所具有的成员变量和方法
  4. 在运行时调用任意一个对象的成员变量和方法
  5. 生成动态代理

1.4.反射相关的主要类

1.java.lang.Class:代表一个类,Class对象表示某个类加载后在堆中的对象
2.java.lang.reflect.Method:代表类的方法,Method对象表示某个类的方法
3.java.lang.reflect.Field:代表类的成员变量,Field对象表示某个类的成员变量
4.java.lang.reflect.Constructor:代表类的构造方法,Constructor对象表示构造器

这些类在java.lang.reflection

代码演示:

public class Reflection01 {
    public static void main(String[] args) throws Exception {

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

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

        //java.lang.reflect.Field:代表类的成员变量,Field对象表示某个类的成员变量
        //得到name字段
        //getFiled不能得到私有的属性
        Field nameField = cls.getField("age");
        System.out.println(nameField.get(o)); // 传统写法 对象.成员变量,反射:成员变量对象.get(对象)

        //java.lang.reflect.Constructor:代表类的构造方法,Constructor对象表示构造器
        Constructor constructor = cls.getConstructor();//()中可以指定构造参数类型,返回无参构造器
        System.out.println(constructor);//Cat()

        Constructor constructor1 = cls.getConstructor(String.class);//传入的 String.class 就是String类的Class对象
        System.out.println(constructor1);//Cat(String name)
    }

1.5.反射优点和缺点

1.优点:可以动态的创建和使用对象(也是框架底层核心),使用灵活,没有反射机制,框架技术就失去底层支撑。
2.缺点:使用反射基本是解释执行,对执行速度有影响。

反射调用优化-关闭访问检查
1.Method和Field、Constructor对象都有setAccessible()方法
2.setAccessible作用是启动和禁用访问安全检查的开关
3.参数值为true表示 反射的对象在使用时取消访问检查,提高反射的效率。参数值为false 则表示反射的对象执行访问检查。

代码:

/**
 * 测试反射调用的性能,和优化方案
 */
public class Reflection02 {
    public static void main(String[] args) throws Exception {
        m1();
        m2();
        m3();
    }

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

    //反射机制调用方法hi
    public static void m2() throws Exception {
        Class cls = Class.forName("com.hspedu.Cat");
        Object o = cls.newInstance();
        Method methodName = cls.getMethod("hi");
        long start = System.currentTimeMillis();
        for (int i = 0; i < 900000000; i++) {
            methodName.invoke(o);//反射调用方法
        }
        long end = System.currentTimeMillis();
        System.out.println("m2 耗时=" + (end - start));
    }

    //反射调用优化+关闭访问检查
    public static void m3() throws Exception {
        Class cls = Class.forName("com.hspedu.Cat");
        Object o = cls.newInstance();
        Method methodName = cls.getMethod("hi");
        methodName.setAccessible(true);//取消在反射调用方法时,取消访问检查
        long start = System.currentTimeMillis();
        for (int i = 0; i < 900000000; i++) {
            methodName.invoke(o);//反射调用方法
        }
        long end = System.currentTimeMillis();
        System.out.println("m3 耗时=" + (end - start));
    }
}

2.Class类

2.1.基本介绍

1.Class也是类,因此也继承Object类

2.Class类对象不是new出来的,而是系统创建的

        //(1).传统new对象
        /**
         * ClassLoader类
         public Class<?> loadClass(String name) throws ClassNotFoundException {
            return loadClass(name, false);
         }
         */
        //Cat cat = new Cat();
        //(2).反射方式,没有debug到 ClassLoader类的 loadClass,原因是:没有注销Cat cat = new Cat();
        /**
         *
         * ClassLoader类,仍然是通过 ClassLoader类加载Cat类的 Class对象
         public Class<?> loadClass(String name) throws ClassNotFoundException {
            return loadClass(name, false);
         }
         */
        Class aClass = Class.forName("com.hspedu.Cat");

3.对于某个类的Class类对象,在内存中只有一份,因为类只加载一次

        Class cls1 = Class.forName("com.hspedu.Cat");

        //3.对于某个类的Class类对象,在内存中只有一份,因为类只加载一次
        Class cls2 = Class.forName("com.hspedu.Cat");
        System.out.println(cls1.hashCode());//1163157884
        System.out.println(cls2.hashCode());//1163157884
        Class cls3 = Class.forName("com.hspedu.Dog");
        System.out.println(cls3.hashCode());//1956725890

4.每个类的实例都会记得自己是由哪个Class实例所生成
5.通过Class可以完整地得到一个类的完整结构,通过一系列API
6.Class对象是存放在堆的
7.类的字节码二进制数据,是放在方法区的,有的地方称为类的元数据(包括方法代码,变量名,方法名,访问权限等等)

2.2.Class类的常用方法

方法名 功能说明
static Class forName(String name) 返回指定类名 name 的 Class对象
Object newInstance() 调用缺省构造函数,返回该Class对象的一个实例
getName() 返回此Class对象所表示的实体(类、接口、数组类、基本类型等)名称
Class[] getInterfaces 获取当前Class对象的接口
ClassLoader getClassLoader() 返回该类的类加载器
Class getSuperclass() 返回表示此Class所表示的实体的超类的Class
Constructor[] getConstructors 返回一个包含某些Constructor对象的数组
Field[] getDeclaredFields() 返回Field对象的一个数组
Method getMethod(String name, Class ... paramTypes 返回一个Method对象,此对象的形参类型为paramType

Class实例演示:

        String classAllPath = "com.hspedu.Car";
        //1.获取到Cart类 对应的 Class对象
        //<?> 表示不确定的Java类型
        Class<?> cls = Class.forName(classAllPath);
        //2.输出cls
        System.out.println(cls);//显示cls对象,是哪个类的Class对象 com.hspedu.Car
        System.out.println(cls.getClass());//输出cls运行类型 java.lang.Class
        //3.得到包名
        System.out.println(cls.getPackage().getName());//包名
        //4.得到全类名
        System.out.println(cls.getName());
        //5.通过cls创建对象实例
        Car car = (Car) cls.newInstance();
        System.out.println(car);//car.toString()
        //6.通过反射获取属性 brand
        Field brand = cls.getField("brand");
        System.out.println(brand.get(car));//宝马
        //7.通过反射给属性赋值
        brand.set(car, "奔驰");
        System.out.println(brand.get(car));//奔驰
        //8.希望可以得到所有的属性(字段)
        System.out.println("===所有的字段属性===");
        Field[] fields = cls.getFields();
        for (Field field : fields) {
            System.out.println(field.getName());//名称
        }

2.3.获取Class类对象

1.前提:已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,可能抛出ClassNotFoundException,实例: Class cls1 = Class.forName("java.lang.Cat");
应用场景:多用于配置文件,读取类全路径,加载类。
2.前提:若已知具体的类,通过类的class获取,该方式最为安全可靠,程序性能最高,实例:Class cls2 = Cat.class;
应用场景:多用于参数传递,比如通过反射得到对应构造器对象。
3.前提:已知某个类的实例,调用该实例的getClass()方法获得Class对象,实例:Class clazz = 对象.getClass();//运行类型
应用场景:通过创建好的对象,获取Class对象
4.其它方式
ClassLoader cl = 对象.getClass().getClassLoader();
Class clazz4 = cl.loadClass("类的全类名");
5.基本数据(int,char,boolean,float,double,byte,long,short)按如下方式得到Class类对象
Class cls = 基本数据类型.class
6.基本数据类型对应的包装类,可以通过.TYPE得到Class类对象
Class cls = 包装类.TYPE

代码演示:

        //1.Class.forName
        String classAllPath = "com.hspedu.Car";//通过读取配置文件读取
        Class<?> cls1 = Class.forName(classAllPath);
        System.out.println(cls1);

        //2.类名.class,应用场景:用于参数传递
        Class cls2 = Car.class;
        System.out.println(cls2);

        //3.对象.getClass(),应用场景:有对象实例
        Car car = new Car();
        Class 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(cls3);

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

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

        System.out.println(integerClass.hashCode());//1956725890
        System.out.println(type1.hashCode());//1956725890

2.4.哪些类型有Class对象

1.外部类,成员内部类,静态内部类,局部内部类,匿名内部类
2.interface:接口
3.数组
4.enum:枚举
5.annotation:注解
6.基本数据类型
7.void

演示哪些类型有Class对象:

        Class<String> cls1 = String.class;//外部类
        Class<Serializable> cls2 = Serializable.class;//接口
        Class<Integer[]> cls3 = Integer[].class;//数组
        Class<float[][]> cls4 = float[][].class;//二维数组
        Class<Deprecated> cls5 = Deprecated.class;//注解
        //枚举
        Class<Thread.State> cls6 = Thread.State.class;
        Class<Long> cls7 = Long.class;//基本数据类型
        Class<Void> cls8 = void.class;//void数据类型
        Class<Class> cls9 = Class.class;

        System.out.println(cls1);
        System.out.println(cls2);
        System.out.println(cls3);
        System.out.println(cls4);
        System.out.println(cls5);
        System.out.println(cls6);
        System.out.println(cls7);
        System.out.println(cls8);
        System.out.println(cls9);

3.类加载

3.1.动态和静态加载

反射机制是Java实现动态语言的关键,也就是通过反射实现类动态加载。
1.静态加载:编译时加载相关的类,如果没有则报错,依赖性太强
2.动态加载:运行时加载需要的类,如果运行时不用该类,即使不存在该类,则不报错,降低了依赖性。

代码演示:

import java.util.*;
import java.lang.reflect.*;

public class ClassLoad_ {
    public static void main(String[] args) throws Exception {
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入key");
        String key = scanner.next();
        switch (key) {
            case "1":
                Dog dog = new Dog();//静态加载,依赖性很强
                dog.cry();
                break;
            case "2":
                //反射-> 动态加载
                Class cls = Class.forName("Person");//加载Person类[动态加载]
                Object o = cls.newInstance();
                Method method = cls.getMethod("hi");
                method.invoke(o);
                System.out.println("ok");
            default:
                System.out.println("do nothing...");
        }
    }
}

//因为new Dog() 是静态加载,因此必须编写Dog
//Person类是动态加载,所以,没有编写Person也不会报错,只有当动态加载该类时,才会报错
class Dog {
    public void cry() {
        System.out.println("小狗汪汪叫");
    }
}

class Person {
	public void hi() {
		System.out.println("小孩 hi");
	}
}

3.2.流程图

  • 类加载流程图

  • 类加载各阶段完成任务

验证:对文件的安全性进行校验,如文件描述符、字节码文件元数据
准备:对静态变量进行默认初始化,并分配空间
解析:把符号引用转成直接引用
加载和连接阶段,其实是由JVM来控制,我们其实是无法控制的;初始化阶段,才是程序员可以指定的。

3.3.类加载五个阶段

  • 加载阶段

JVM在该阶段的主要目的是将字节码从不同的数据源(可能是class文件、也可能是jar包,甚至网络)转化为二进制字节流加载到内存中,并生成一个代表该类的java.lang.Class对象

  • 连接阶段-验证

1.目的是为了确保Class文件的字节流种包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。
2.包括:文件格式验证(是否以魔数 oxcafebabe 开头)、元数据验证、字节码验证和符号引用验证
3.可以考虑使用 -Xverify:none 参数来关闭大部分的类验证措施,缩短虚拟机类加载的时间。

  • 连接阶段-准备

1.JVM会在该阶段对静态变量,分配内存并初始化(对应数据类型的默认初始值,如 0、0L、null、false等)。这些变量所使用的内存都将在方法区中进行分配
2.说明一个类加载的连接阶段-准备

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;
}

  • 连接阶段-解析

1.虚拟机将常量池内的符号引用替换为直接引用的过程
2.举例说明:
A类引用B类,解析前是相当于是一个相对的关系,没有内存地址的概念,解析后是A类和B类的相对关系,就使用替换内存地址的实际引用

  • 初始化阶段

1.到初始化阶段,才真正开始执行类中定义的 Java 程序代码,此阶段是执行<clinit>()方法的过程。
2.clinit()方法是由编译器按语句在源文件中出现的顺序,依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并进行合并。

public class ClassLoad03 {
    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
         */
        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() 构造器被执行");
    }
}

运行结果:

3.虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确地加锁、同步,如果多个线程同时去初始化一个类,那么只会由一个线程去执行这个类的<clinit>()方法,其他线程都需要阻塞等待,直到活动线程执行<clinit>()方法完毕。

        //加载类的时候,是有同步机制控制
        /*
            protected Class<?> loadClass(String name, boolean resolve)
                throws ClassNotFoundException
            {
                //正因为有这个机制,才能保证某个类在内存中,只有一份Class对象
                synchronized (getClassLoadingLock(name)) {
                //......
                }
            }
         */
        B b = new B();

4.反射获取类的结构信息

4.1.Class类

1.getName:获取全类名
2.getSimpleName:获取简单类名
3.getFields:获取所有public修饰的属性,包含本类以及父类的
4.getDeclaredFields:获取本类中所有属性
5.getMethods:获取所有public修饰的方法,包含本类以及父类的
6.getDeclaredMethods:获取本类中所有方法
7.getConstructors:获取本类所有public修饰的构造器
8.getDeclaredConstructors:获取本类中所有构造器
9.getPackage:以Package形式返回 包信息
10.getSuperClass:以Class形式返回父类信息
11.getInterfaces:以Class[]形式返回接口信息
12.getAnnotations:以Annotation[]形式返回注解信息

代码:

import org.junit.Test;

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

/**
 * 演示如何通过反射获取类的结构信息
 */
public class ReflectionUtils {

    //第一组方法API
    @Test
    public void api_01() throws ClassNotFoundException {
        //得到Class对象
        Class personCls = Class.forName("com.hspedu.reflection.Person");
        //1.getName:获取全类名
        System.out.println(personCls.getName());
        //2.getSimpleName:获取简单类名
        System.out.println(personCls.getSimpleName());
        //3.getFields:获取所有public修饰的属性,包含本类以及父类的
        for (Field field : personCls.getFields()) {
            System.out.println("本类以及父类的属性=" + field.getName());
        }
        //4.getDeclaredFields:获取本类中所有属性
        for (Field declaredField : personCls.getDeclaredFields()) {
            System.out.println("本类中所有属性" + declaredField.getName());
        }
        //5.getMethods:获取所有public修饰的方法,包含本类以及父类的
        for (Method method : personCls.getMethods()) {
            System.out.println("本类以及父类的方法=" + method.getName());
        }
        //6.getDeclaredFields:获取本类中所有方法
        for (Method declaredField : personCls.getDeclaredMethods()) {
            System.out.println("本类中所有方法=" + declaredField.getName());
        }
        //7.getConstructors:获取本类所有public修饰的构造器
        for (Constructor constructor : personCls.getConstructors()) {
            System.out.println("本类的构造器=" + constructor);
        }
        //8.getDeclaredConstructors:获取本类中所有构造器
        for (Constructor declaredConstructor : personCls.getDeclaredConstructors()) {
            System.out.println("本类中所有构造器=" + declaredConstructor);
        }
        //9.getPackage:以Package形式返回 包信息
        System.out.println(personCls.getPackage());//com.hspedu.reflection
        //10.getSuperClass:以Class形式返回父类信息
        System.out.println("父类的Class对象=" + personCls.getSuperclass());
        //11.getInterfaces:以Class[]形式返回接口信息
        for (Class anInterface : personCls.getInterfaces()) {
            System.out.println("接口信息=" + anInterface);
        }
        //12.getAnnotations:以Annotation[]形式返回注解信息
        for (Annotation annotation : personCls.getAnnotations()) {
            System.out.println("注解信息=" + annotation);
        }
    }
}

class A {
    public String hobby;

    public void hi() {
    }

    public A() {
    }

}

interface IA {

}

interface IB {

}

@Deprecated
class Person extends A implements IA, IB {
    //属性
    public String name;
    protected int age;
    String job;
    private double sal;

    //构造器
    public Person() {

    }

    public Person(String name) {
    }

    private Person(String name, int age) {
    }

    //方法
    public void m1() {

    }

    protected void m2() {

    }

    void m3() {

    }

    private void m4() {

    }
}

4.2.Field类

1.getModifiers:以int形式返回修饰符
【说明:默认修饰符 是0,public 是1,private 是2,protected 是4,static 是8,final 是16】
2.getType:以Class形式返回类型
3.getName:返回属性名

代码:

    @Test
    public void api_02() throws ClassNotFoundException {
        //得到Class对象
        Class personCls = Class.forName("com.hspedu.reflection.Person");
        //4.getDeclaredFields:获取本类中所有属性
        //【说明:默认修饰符 是0,public 是1,private 是2,protected 是4,static 是8,final 是16】
        for (Field declaredField : personCls.getDeclaredFields()) {
            System.out.println("本类中所有属性" + declaredField.getName()
                    + " 该属性的修饰符值=" + declaredField.getModifiers()
                    + " 该属性的类型=" + declaredField.getType());
        }
    }

4.3.Method类

1.getModifiers:以int形式返回修饰符
【说明:默认修饰符 是0,public 是1,private 是2,protected 是4,static 是8,final 是16】
2.getReturnType:以Class形式获取 返回类型
3.getName:返回方法名
4.getParameterTypes:以Class[]返回参数类型数组

代码:

        //6.getDeclaredFields:获取本类中所有方法
        for (Method declaredField : personCls.getDeclaredMethods()) {
            System.out.println("本类中所有方法=" + declaredField.getName()
                    + " 该方法的访问修饰符值=" + declaredField.getModifiers()
                    + " 该方法的返回类型=" + declaredField.getReturnType());
            //输出当前方法的形参数组情况
            for (Class<?> parameterType : declaredField.getParameterTypes()) {
                System.out.println("该方法的形参类型" + parameterType);
            }
        }

4.4.Constructor类

1.getModifiers:以int形式返回修饰符
2.getName:返回构造器名(全类名)
3.getParameterTypes:以Class[]返回参数类型数组

代码:

        //8.getDeclaredConstructors:获取本类中所有构造器
        for (Constructor declaredConstructor : personCls.getDeclaredConstructors()) {
            System.out.println("本类中所有构造器=" + declaredConstructor);
            for (Class parameterType : declaredConstructor.getParameterTypes()) {
                System.out.println("该构造器的形参类型" + parameterType);
            }
        }

4.5.反射暴破创建实例

1.方式一:调用类中的public修饰的无参构造器
2.方式二:调用类中的指定构造器

3.Class类相关方法:

  • newInstance:调用类中的无参构造器,获取对应类的对象
  • getConstructor(Class... clazz):根据参数列表,获取对应的public构造器对象
  • getDeclaredConstructor(Class... clazz):根据参数列表,获取对应的所有构造器对象

4.Constructor类相关方法

  • setAccessible:暴破
  • newInstance(Object... obj):调用构造器

案例演示:
1.测试1:通过反射创建某类的对象,要求该类中必须有public的无参构造
2.测试2:通过调用某个特定构造器的方式,实现创建某类的对象

代码:

/**
 * 演示通过反射机制创建实例
 */
public class ReflectCreateInstance {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {

        //1.先获取到User类的Class对象
        Class userClass = Class.forName("com.hspedu.reflection.User");

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

        //3.通过public的有参构造器创建实例
        /*
            constructor 对象就是
            public User(String name) {//public的有参构造器
                this.name = name;
            }
         */
        //3.1.先得到对应构造器
        Constructor constructor = userClass.getConstructor(String.class);
        //3.2.创建实例,并传入实参
        Object zs = constructor.newInstance("张三");
        System.out.println("zs=" + zs);

        //4.通过非public的有参构造器创建实例
        //4.1.得到private的构造器对象
        Constructor declaredConstructor = userClass.getDeclaredConstructor(int.class, String.class);
        //4.2.创建实例
        //暴破【暴力破解】,使用反射可以访问private构造器、方法、属性,反射面前,都是纸老虎
        declaredConstructor.setAccessible(true);
        Object kisen = declaredConstructor.newInstance(18, "kisen");
        System.out.println("kisen=" + kisen);
    }
}

class User {
    private int age = 10;
    private String name = "冰枫";

    public User() {//无参 public

    }

    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 + '\'' +
                '}';
    }
}

4.6.反射暴破操作属性

1.根据属性名获取Field对象
Field f = clazz对象.getDeclaredField(属性名);
2.暴破:f.setAccessible(true); //f 是Field
3.访问
f.set(o, 值); //o 表示对象
syso(f.get(o)); //o 表示对象
4.注意:如果是静态属性,则set和get中的参数o,可以写成null
5.案例演示

/**
 * 演示反射操作属性
 */
public class ReflectAccessProperty {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchFieldException {

        //1.得到Student类对应的Class对象
        Class<?> stuClass = Class.forName("com.hspedu.reflection.Student");
        //2.创建对象
        Object o = stuClass.newInstance(); //o 的运行类型就是Student
        System.out.println(o.getClass()); //Student
        //3.使用反射得到age 属性对象
        Field age = stuClass.getField("age");
        age.set(o, 88);//通过反射来操作属性
        System.out.println(o);
        System.out.println(age.get(o));//返回age属性的值

        //4.使用反射操作name 属性
        Field name = stuClass.getDeclaredField("name");
        //对name 进行暴破,可以操作private 属性
        name.setAccessible(true);
        //name.set(o, "冰枫");
        name.set(null, "冰枫");//因为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 + '\'' +
                '}';
    }
}

4.7.反射暴破操作方法

1.根据方法名和参数列表获取Method方法对象:Method m = clazz.getDeclaredMethod(方法名,XXX.class); //得到本类的所有方法
2.获取对象:Object o = clazz.newInstance();
3.暴破:m.setAccessible(true);
4.访问:Object returnValue = m.invoke(o, 实参列表); //o 就是对象
5.注意:如果是静态方法,则invoke的参数o,可以写成null

/**
 * 演示通过反射调用方法
 */
public class ReflectAccessMethod {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {

        //1.得到Boss类对应的Class对象
        Class<?> bossCls = Class.forName("com.hspedu.reflection.Boss");
        //创建对象
        Object o = bossCls.newInstance();
        //3.调用public的hi方法
        //Method hi = bossCls.getMethod("hi", String.class);
        //3.1.得到hi方法对象
        Method hi = bossCls.getDeclaredMethod("hi", String.class);
        //3.2.调用
        hi.invoke(o, "Lucy");

        //4.调用private static 方法
        //4.1.得到 say 方法对象
        Method say = bossCls.getDeclaredMethod("say", int.class, String.class, char.class);
        //4.2.因为say方法是private,所以需要暴破,原理和前面讲的构造器和属性一样
        say.setAccessible(true);
        System.out.println(say.invoke(o, 100, "Marry", '女'));
        //4.3.因为say方法是statitc的,还可以这样调用,可以传入null
        System.out.println(say.invoke(null, 200, "Jessy", '女'));

        //5.在反射中,如果方法有返回值,统一返回Object,但是它运行类型和方法定义的返回类型一致
        Object retVal = say.invoke(null, 300, "Rose", '男');
        System.out.println("retVal 的运行类型=" + retVal.getClass());//String
    }
}

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

    public Boss() {//构造器

    }

    private static String say(int n, String s, char c) {//静态方法
        return n + " " + s + " " + c;
    }

    public void hi(String s) {//普通public方法
        System.out.println("hi " + s);
    }
}

4.8.练习

练习1:通过反射修改私有成员变量
1.定义PrivateTest类,有私有name属性,并且属性值为hellokitty
2.提供getName的公有方法
3.创建PrivateTest的类,利用Class类得到私有的name属性,修改私有的name属性值,并调用getName()的方法打印name属性值

代码实现:

public class Homework01 {
    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        //1.得到PrivateTest类对应的Class对象
        Class<?> privateTestClass = PrivateTest.class;
        //2.创建对象实例
        Object privateTestObj = privateTestClass.newInstance();
        //3.得到name属性对象
        Field name = privateTestClass.getDeclaredField("name");
        //4.暴破
        name.setAccessible(true);
        System.out.println(name.get(privateTestObj));
        name.set(privateTestObj, "helloreflect");

        //5.得到getName方法对象
        Method getName = privateTestClass.getMethod("getName");
        //6.因为getName()是public,所以直接调用
        System.out.println("name属性值=" + getName.invoke(privateTestObj));

    }
}

class PrivateTest {
    private String name = "hellokitty";

    public String getName() {
        return name;
    }
}

练习2:利用反射和File完成以下功能
1.利用Class类的forName方法得到File类的class 对象
2.在控制台打印File类的所有构造器
3.通过newInstance的方法创建File对象,并创建E:\mynew.txt文件
提示:创建文件的正常写法如下:
File file = new File("d:\aa.txt");//内存
file.createNewFile();//方法,才能真正的创建文件

代码实现:

public class Homework02 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, IOException {
        //1.利用Class类的forName方法得到File类的class 对象
        Class<?> fileClass = Class.forName("java.io.File");
        //2.得到所有的构造器
        for (Constructor<?> declaredConstructor : fileClass.getDeclaredConstructors()) {
            System.out.println(declaredConstructor);
        }
        //3.指定的得到public java.io.File(java.lang.String)
        Constructor<?> declaredConstructor = fileClass.getDeclaredConstructor(String.class);
        String fileAllPath = "d:\\mynew.txt";
        Object file = declaredConstructor.newInstance(fileAllPath);//创建File对象

        //4.得到createNewFile 的方法对象
        Method createNewFile = fileClass.getMethod("createNewFile");
        createNewFile.invoke(file);//创建文件,调用的是 createNewFile
        //file的运行类型就是File
        System.out.println(file.getClass());
        System.out.println("创建文件成功" + fileAllPath);
    }
}
posted @ 2022-01-07 11:18  冰枫丶  阅读(82)  评论(0编辑  收藏  举报