1、什么是反射
反射允许对封装类的字段,方法和构造函数的信息进行编程访问。
也就是:
反射允许对成员变量,成员方法和构造方法的信息进行编程访问。
2、获取class对象
获取一个类的字节码文件对象:
方式1:Class.forName("全类名"); 此方式最常用
Class类是定义好的,用来描述字节码文件。括号里传递一个类的全类名。
适用于源代码阶段:比如:A.java文件编译成A.class文件,没有把代码加载到内存中,只是在硬盘中进行操作。
方式2:类名.class 此方式一般是当做参数进行传递,如synchronized方法中的锁对象
适用于加载阶段:把A.class文件加载到内存中时
方式3:对象.getClass(); 当有了这个类的对象时,才可以使用此方式
getClass()方法定义在Object类中,所有对象都可以调用
适用于运行阶段:在内存中 A a=new A(); 创建这个类的对象,通过对象获取class字节码文件对象
3、反射获取构造方法
通过Constructor类,该类用来描述构造方法,该类对象就表示构造方法的对象。
public class MyReflectDemo {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {
    /*
    Class类中用于获取构造方法的方法
      Constructor<?>[] getConstructors()                                返回所有公共构造方法对象的数组
      Constructor<?>[] getDeclaredConstructors()                        返回所有构造方法对象的数组
      Constructor<T> getConstructor(Class<?>... parameterTypes)         返回单个公共构造方法对象
      Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) 返回单个构造方法对象
    Constructor类中用于创建对象的方法
      T newInstance(Object... initargs)                   根据指定的构造方法创建对象
      setAccessible(boolean flag)                         设置为true,表示取消访问检查
    */
        //1.获取class字节码文件对象
        Class clazz = Class.forName("com.itheima.myreflect2.Student");
        System.out.println(clazz);//class com.itheima.myreflect2.Student
       
        //2.获取构造方法
        Constructor[] cons1 = clazz.getConstructors();
        for (Constructor con : cons1) {
            System.out.println(con);
        }
        /*  运行结果:
            public com.itheima.myreflect2.Student()
            public com.itheima.myreflect2.Student(java.lang.String)*/
        
     
        Constructor[] cons2 = clazz.getDeclaredConstructors();
        for (Constructor con : cons2) {
            System.out.println(con);
        }
        /*  运行结果:
            public com.itheima.myreflect2.Student()
            public com.itheima.myreflect2.Student(java.lang.String)
            protected com.itheima.myreflect2.Student(int)
            private com.itheima.myreflect2.Student(java.lang.String,int)
*/
        Constructor con1 = clazz.getDeclaredConstructor();
        System.out.println(con1);//public com.itheima.myreflect2.Student()
        
        
        Constructor con2 = clazz.getDeclaredConstructor(String.class);
        System.out.println(con2);//public com.itheima.myreflect2.Student(java.lang.String)
        
        
        Constructor con3 = clazz.getDeclaredConstructor(int.class);
        System.out.println(con3);//protected com.itheima.myreflect2.Student(int)
        
        
        
        Constructor con4 = clazz.getDeclaredConstructor(String.class,int.class);
        int modifiers = con4.getModifiers();//获取构造方法的权限修饰符
        System.out.println(modifiers);//2    (1:public,2:private)
       
        Parameter[] parameters = con4.getParameters();//获取构造方法所有的参数
        for (Parameter parameter : parameters) {
            System.out.println(parameter);
        }
        /*  运行结果:
            java.lang.String arg0
            int arg1
        */
        
        //暴力反射:表示临时取消权限校验,可以访问私有的了
        con4.setAccessible(true);
        Student stu = (Student) con4.newInstance("张三", 23);
        System.out.println(stu);//Student{name = 张三, age = 23}
    }
} 
Student类:
public class Student {
    private String name;
    private int age;
    public Student() {
    }
    public Student(String name) {
        this.name = name;
    }
    protected Student(int age) {
        this.age = age;
    }
    private Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    public String toString() {
        return "Student{name = " + name + ", age = " + age + "}";
    }
} 
 
4、反射获取成员变量
通过Field类获取
public class MyReflectDemo {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException {
    /*
       Class类中用于获取成员变量的方法
            Field[] getFields():                返回所有公共成员变量对象的数组
            Field[] getDeclaredFields():        返回所有成员变量对象的数组
            Field getField(String name):        返回单个公共成员变量对象
            Field getDeclaredField(String name):返回单个成员变量对象
       Field类中用于创建对象的方法
            void set(Object obj, Object value):赋值
            Object get(Object obj)              获取值
    */
        //1.获取class字节码文件的对象
        Class clazz = Class.forName("com.itheima.myreflect3.Student");
//        Field[] fields = clazz.getFields();//获取公共的成员变量
        //2.获取所有的成员变量,私有的也能获取到
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            System.out.println(field);
        }
        //获取单个的成员变量,获取私有的
        Field name = clazz.getDeclaredField("name");
        System.out.println(name);
        //获取权限修饰符
        int modifiers = name.getModifiers();
        System.out.println(modifiers);
        //获取成员变量的名字
        String n = name.getName();
        System.out.println(n);
        //获取成员变量的数据类型
        Class<?> type = name.getType();
        System.out.println(type);
        //获取成员变量记录的值,跟对象有关系,先获取对象
        Student s = new Student("zhangsan",23,"男");
        name.setAccessible(true);
        String value = (String) name.get(s);
        System.out.println(value);
        //修改对象里面记录的值
        name.set(s,"lisi");
        System.out.println(s);
    }
} 
public class Student {
    private String name;
    private int age;
    public String gender;
    public Student() {
    }
    public Student(String name, int age, String gender) {
        this.name = name;
        this.age = age;
        this.gender = gender;
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    /**
     * 获取
     * @return gender
     */
    public String getGender() {
        return gender;
    }
    /**
     * 设置
     * @param gender
     */
    public void setGender(String gender) {
        this.gender = gender;
    }
    public String toString() {
        return "Student{name = " + name + ", age = " + age + ", gender = " + gender + "}";
    }
} 
5、反射获取成员方法
通过Method类
public class MyReflectDemo {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {
    /*
       Class类中用于获取成员方法的方法
            Method[] getMethods():返回所有公共成员方法对象的数组,包括继承的
            Method[] getDeclaredMethods():返回所有成员方法对象的数组,不包括继承的
            Method getMethod(String name, Class<?>... parameterTypes) :返回单个公共成员方法对象
            Method getDeclaredMethod(String name, Class<?>... parameterTypes):返回单个成员方法对象
       Method类中用于创建对象的方法
            Object invoke(Object obj, Object... args):运行方法
            参数一:用obj对象调用该方法
            参数二:调用方法的传递的参数(如果没有就不写)
            返回值:方法的返回值(如果没有就不写)
        获取方法的修饰符
        获取方法的名字
        获取方法的形参
        获取方法的返回值
        获取方法的抛出的异常
    */
        
        //1. 获取class字节码文件对象
        Class clazz = Class.forName("com.itheima.myreflect4.Student");
        //2. 获取里面所有的方法对象(包含父类中所有的公共方法)
        Method[] methods = clazz.getMethods();
        for (Method method : methods) {
            System.out.println(method);
        }
        // 获取里面所有的方法对象(不能获取父类的,但是可以获取本类中私有的方法)
        Method[] methods1 = clazz.getDeclaredMethods();
        for (Method method : methods1) {
            System.out.println(method);
        }
        // 获取指定的单一方法
        Method m = clazz.getDeclaredMethod("eat", String.class);//方法的名字,方法的形参
        System.out.println(m);
        // 获取方法的修饰符
        int modifiers = m.getModifiers();
        System.out.println(modifiers);
        // 获取方法的名字
        String name = m.getName();
        System.out.println(name);
        // 获取方法的形参
        Parameter[] parameters = m.getParameters();
        for (Parameter parameter : parameters) {
            System.out.println(parameter);
        }
        //获取方法的抛出的异常
        Class[] exceptionTypes = m.getExceptionTypes();
        for (Class exceptionType : exceptionTypes) {
            System.out.println(exceptionType);
        }
        //方法运行
        /*Method类中用于创建对象的方法
        Object invoke(Object obj, Object... args):运行方法
        参数一:用obj对象调用该方法
        参数二:调用方法的传递的参数(如果没有就不写)
        返回值:方法的返回值(如果没有就不写)*/
        
        Student s = new Student();
        m.setAccessible(true);
        //参数一s:表示方法的调用者
        //参数二"汉堡包":表示在调用方法的时候传递的实际参数
        String result = (String) m.invoke(s, "汉堡包");
        System.out.println(result);
    }
} 
public class Student {
    private String name;
    private int age;
    public Student() {
    }
    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    public void sleep(){
        System.out.println("睡觉");
    }
    private String eat(String something) throws IOException,NullPointerException,ClassCastException {
        System.out.println("在吃" + something);
        return "奥利给";
    }
    private void eat(String something,int a) {
        System.out.println("在吃" + something);
    }
    public String toString() {
        return "Student{name = " + name + ", age = " + age + "}";
    }
} 
6、反射的作用
作用1:获取一个类里面所有的信息,获取到了之后,再执行其他的业务逻辑
作用2:结合配置文件,动态的创建对象并调用方法
练习1:保存任意对象的数据
public class MyReflectDemo {
    public static void main(String[] args) throws IllegalAccessException, IOException {
    /*
        对于任意一个对象,都可以把对象所有的字段名和值,保存到文件中去
    */
       Student s = new Student("小A",23,'女',167.5,"睡觉");
       Teacher t = new Teacher("播妞",10000);
       saveObject(s);
    }
    //把对象里面所有的成员变量名和值保存到本地文件中
    public static void saveObject(Object obj) throws IllegalAccessException, IOException {
        //1.获取字节码文件的对象
        Class clazz = obj.getClass();
        //2. 创建IO流
        BufferedWriter bw = new BufferedWriter(new FileWriter("myreflect\\a.txt"));
        //3. 获取所有的成员变量
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            field.setAccessible(true);
            //获取成员变量的名字
            String name = field.getName();
            //获取成员变量的值
            Object value = field.get(obj);
            //写出数据
            bw.write(name + "=" + value);
            bw.newLine();
        }
        bw.close();
    }
} 
public class Student {
    private String name;
    private int age;
    private char gender;
    private double height;
    private String hobby;
    public Student() {
    }
    public Student(String name, int age, char gender, double height, String hobby) {
        this.name = name;
        this.age = age;
        this.gender = gender;
        this.height = height;
        this.hobby = hobby;
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    /**
     * 获取
     * @return gender
     */
    public char getGender() {
        return gender;
    }
    /**
     * 设置
     * @param gender
     */
    public void setGender(char gender) {
        this.gender = gender;
    }
    /**
     * 获取
     * @return height
     */
    public double getHeight() {
        return height;
    }
    /**
     * 设置
     * @param height
     */
    public void setHeight(double height) {
        this.height = height;
    }
    /**
     * 获取
     * @return hobby
     */
    public String getHobby() {
        return hobby;
    }
    /**
     * 设置
     * @param hobby
     */
    public void setHobby(String hobby) {
        this.hobby = hobby;
    }
    public String toString() {
        return "Student{name = " + name + ", age = " + age + ", gender = " + gender + ", height = " + height + ", hobby = " + hobby + "}";
    }
} 
public class Teacher {
    private String name;
    private double salary;
    public Teacher() {
    }
    public Teacher(String name, double salary) {
        this.name = name;
        this.salary = salary;
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return salary
     */
    public double getSalary() {
        return salary;
    }
    /**
     * 设置
     * @param salary
     */
    public void setSalary(double salary) {
        this.salary = salary;
    }
    public String toString() {
        return "Teacher{name = " + name + ", salary = " + salary + "}";
    }
} 
练习2:利用反射动态的创建对象和运行方法
public class MyReflectDemo {
    public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {
    /*
        反射可以跟配置文件结合的方式,动态的创建对象,并调用方法
    */
        //1.读取配置文件中的信息
        Properties prop = new Properties();
        FileInputStream fis = new FileInputStream("myreflect\\prop.properties");
        prop.load(fis);
        fis.close();
        System.out.println(prop);
        //2.获取全类名和方法名
        String className = (String) prop.get("classname");
        String methodName = (String) prop.get("method");
        System.out.println(className);
        System.out.println(methodName);
        //3.利用反射创建对象并运行方法
        Class clazz = Class.forName(className);
        //获取构造方法
        Constructor con = clazz.getDeclaredConstructor();
        Object o = con.newInstance();
        System.out.println(o);
        //获取成员方法并运行
        Method method = clazz.getDeclaredMethod(methodName);
        method.setAccessible(true);
        method.invoke(o);
    }
} 
package com.itheima.myreflect6;
public class Student {
    private String name;
    private int age;
    public Student() {
    }
    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public void study(){
        System.out.println("学生在学习!");
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    public String toString() {
        return "Student{name = " + name + ", age = " + age + "}";
    }
}
 
package com.itheima.myreflect6;
public class Teacher {
    private String name;
    private double salary;
    public Teacher() {
    }
    public Teacher(String name, double salary) {
        this.name = name;
        this.salary = salary;
    }
    public void teach(){
        System.out.println("老师在教书!");
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return salary
     */
    public double getSalary() {
        return salary;
    }
    /**
     * 设置
     * @param salary
     */
    public void setSalary(double salary) {
        this.salary = salary;
    }
    public String toString() {
        return "Teacher{name = " + name + ", salary = " + salary + "}";
    }
}
 
配置文件里的内容:

想要运行某个类中的方法,直接更改配置文件中的类名和方法名就行了。
















![[论文精读]Active and Semi-Supervised Graph Neural Networks for Graph Classification](https://i-blog.csdnimg.cn/direct/de3aeae6539c41279f4155cab434f100.gif)


