• 反射


    反射

    反射的步骤:
    	获取对应的类对象:注意有泛型
    	Class<?> cls = Class.forName("reflextion.Customer");
    	构造方法:
    		1。使用:Constructor<?>[] declaredConstructors = cls.getDeclaredConstructors();/Constructor<?>[] constructors = cls.getConstructors();获取所有的构造方法,此处需要增加泛型<?>
    		2。使用:Constructor<?> declaredConstructor1 = cls.getDeclaredConstructor(String.class);/Constructor<?> constructor1 = cls.getConstructor();获取对应的构造方法,获取构造方法时需要传入对应的参数类型
    		3。如果是非public的构造方法,使用:declaredConstructor.setAccessible(true);设置为可以访问
    		4。获取对应的对象:Customer customer = (Customer) declaredConstructor.newInstance("hahaha");此处需要强转为所需要的类型,并传入合适的参数
    		
    	成员方法:
    		1。使用Method[] declaredMethods = cls.getDeclaredMethods();/Method[] methods = cls.getMethods();方法获取对应的成员方法集合
    		2。使用Method declaredMethod1 = cls.getDeclaredMethod("setId",int.class);/Method game1 = cls.getMethod("getId");获取对应的成员方法,获取时传入适当的方法名和参数类型
    		3.// 创建实例对象
    		Object obj = cls.getConstructor().newInstance();
    		4.如果非public方法,declaredMethod1.setAccessible(true);
    		5.declaredMethod1.invoke(obj,5);反射而来的方法.invoke在哪个实例对象上面,同时传入合适的参数
    		
    		成员变量:
    		1.使用Field[] ms1 = cls.getFields()/Field[] ms2 = cls.getDelaredFields()获取本身及父类public属性/本身所有属性
    		2.使用Field id = cls.getField("id");获取到反射类型的属性.id为一个对象
    		3.创建实例对象
    		Object o = cls.getConstructor().newInstance();该对象的类型为cls的类型
    		3.5:如果为非public属性,还需要设置setAccessible
    		4。id.set(在哪个实例对象上,赋值)
    		5.id.get(在哪个实例对象上取值)
    		
    		
    

    1. 反射概述

    1.1 Java文件和.class文件的关系
    Java文件
    	Java文件中包含代码的所有内容,类,接口,成员变量,成员方法....
    
    .class字节码问题
    	.java文件 通过 javac编译工具生成对应的.class字节码文件
    	使用JDK中提供的反编译工具,可以看到.class文件中包含 
    		Class 完整的包名.类名
    		Field 成员变量,成员变量的名字和成员变量的数据类型[如果是引用数据类型,也是
    			完整的包名.类名]
    		Method 成员方法,方法权限修饰符,返回值类型,方法名,形式参数列表数据类型
    
    总结:
    	.class字节码文件中,包含了Java文件的所有内容
    

    class1.png

    class2.png

    1.2 程序加载过程和.class文件的关系
    	在Java文件运行过程中,当前程序需要哪一个类参与代码执行,那么就需要加载这个类的.class字节码文件,该.class字节码文件时在程序的加载阶段,存在于内存的【代码区】
    	
    	.class字节码文件既然加载到内存的【代码区】
    	.class文件中包含对应Java程序的所有内容
    	代码区存在一块空间 ==> .class ==> Java程序的所有内容
    
    1.3 Java中的万物皆对象
    	在Java代码中,把在内存代码区保存的.class字节码内存空间,看做是一个对象。而该对象中包含了对应Java文件的所有内容。
    

    class.png

    1.4 Class到底是什么?
    class Person {
        int age;
        String name;
        
        public Person() {}
        
        public Person(int age, String name) {
            this.age = age;
            this.name = name;
        }
        
        public void test() {
            sout("方法");
        }
    }
    
    class Dog {
        String name;
        char gender;
        
        public Dog() {}
        
        public Dog(String name, char gender) {
            this.name = name;
            this.gender = gender;
        }
        
        public void eat() {
            sout("狗狗吃肉");
        }
    }
    

    Class.png

    2. 反射必会方法【重点】

    2.1 Class涉及到的方法
    Class Class.forName(String packageNameAndClassName);
    	Class类的静态成员方法,通过完整的包名.类名获取对应.class文件的Class对象
        同时也可以作为.class文件加载的方法。
    
    Class 类名.class;
        通过类名.class方法,获取对应的Class类对象,通常用于方法的参数类型。
    
    Class 类对象.getClass();
    	通过类对象获取对应.class的Class类对象,方法参数,或者说数据类型判断。
    
    package com.qfedu.a_reflect;
    
    /*
    Class Class.forName(String packageNameAndClassName);    
    	Class类的静态成员方法,通过完整的包名.类名获取对应.class文件的Class对象   
    	同时也可以作为.class文件加载的方式。
    Class 类名.class;    
    	通过类名.class方法,获取对应的Class类对象,通常用于方法的参数类型。
    Class 类对象.getClass();    
    	通过类对象获取对应.class的Class类对象,方法参数,或者说数据类型判断。
     */
    public class GetClassObject {
    	public static void main(String[] args) throws ClassNotFoundException {
    		System.out.println(123456);
    		
    		Class<?> forName = Class.forName("com.qfedu.a_reflect.Person");
    		
    		Class<com.qfedu.a_reflect.Person> cls = Person.class;
    		
    		Class<? extends Person> class1 = new Person().getClass();
    		
    		/*
    		 * 请问这个三个Class对象是不是同一个Class对象???
    		 * 		Class对象对应的是在内存代码区的.class文件占用的内存空间
    		 * 		Class引用数据类型变量保存的就是当前空间首地址,
    		 * 		Java程序中,.class字节码文件有且之加载一次
    		 * 		.class文件占用的空间独一份,不管通过哪一种方式获取对应的Class类对象
    		 * 		都是同一个对象
    		 */
    		System.out.println(forName == cls);
    		System.out.println(class1 == cls);
    		System.out.println(class1 == forName);
    	}
    }
    
    2.2 Constructor 构造方法类涉及到的方法
    public Constructor[] getConstructors();
    	获取当前Class类对象对应Java文件中,所有【public修饰构造方法的类对象数组】
    	
    public Constructor[] getDeclaredConstructors();
    	【暴力反射】
    	获取当前Class类对象对应Java文件中,所有【构造方法的类对象数组】,包括私有化构造方法。
    	
    【回顾】
    	new Person();
    	new Person(1);
    	因为这里利用了重载的知识点,会根据实际【参数类型】,来选择对应的构造方法。
    【推理】
    	通过Class类对象,获取指定构造方法,需要根据构造方法的所需的参数数据类型来完成。
    
    public Constructor getConstructor(Class... initArgumentTypes);
    	根据指定的数据类型,来选择对应的构造方法,这里可能会抛出异常。
    	这里有且只能获取获取类内的指定数据类型public修饰构造方法类对象
    	Class: 约束数据类型,当前方法所需的参数类型
    		例如: 
    			这里需要int类型 int.class
    			这里需要String类型 String.class
    			之类需要Perosn类型 Person.class
    		异常:
    			NoSuchMethodException
    	... : 不定长参数
    		构造方法需要的参数类型是很多的,有可能无参数,有可能有参数。... 不定长参数类约束使用,增强代码的普适性
    		例如:
    			这里无参数 () or (null)
    			参数类型int类型 (int.class)
    			参数类型int, String类型 (int.class, String.class)
    	initArgumentTypes:
    		参数名 初始化参数类型复数
    		
    public Constructor getDeclaredConstructor(Class... initArgumentTypes);
    	【暴力反射】
    	根据指定的数据类型,来选择对应的构造方法,这里可能会抛出异常。
    	这里可以获取指定参数类型私有化构造方法和非私有化构造方法
    	Class: 约束数据类型,当前方法所需的参数类型
    		例如: 
    			这里需要int类型 int.class
    			这里需要String类型 String.class
    			之类需要Perosn类型 Person.class
    		异常:
    			NoSuchMethodException
    	... : 不定长参数
    		构造方法需要的参数类型是很多的,有可能无参数,有可能有参数。... 不定长参数
    		类约束使用,增强代码的普适性
    		例如:
    			这里无参数 () or (null)
    			参数类型int类型 (int.class)
    			参数类型int, String类型 (int.class, String.class)
    	initArgumentTypes:
    		参数名 初始化参数类型复数
    		
    Object newInstance(Object... initArguments);
    	通过Constructor对象来调用,传入当前构造方法所需创建对象的初始化参数,创建对象。
    	Object: Object类是Java中所有类的基类,这里可以传入任意类型的参数
    	... : 不定长参数,因为Constructor类对象在获取的过程中,约束的参数个数都不确定,
    	这里使用不定长参数来传入数据
    
    package reflextion;
    
    import java.lang.reflect.Constructor;
    
    /*
     * 操作Constructor
     */
    public class GetConstructorObject {
    	public static void main(String[] args) throws Exception {
    		/*
    		 * 根据指定的包名.类名,获取对应的Class对象
    		 */
    		Class<?> cls = Class.forName("reflextion.Customer");
    
    		/*
    		 * 获取当前Person类内所有非私有化构造方法
    		 */
    		Constructor<?>[] constructors = cls.getConstructors();
    		for (Constructor<?> constructor : constructors) {
    			System.out.println(constructor);
    		}
    
    		System.out.println("-----获取当前Person类内所有非私有化构造方法------------------------------");
    		System.out.println();
    
    		/*
    		 * 暴力反射,获取Person类内所有的构造方法,包括私有化构造方法
    		 */
    		Constructor<?>[] declaredConstructors = cls.getDeclaredConstructors();
    		for (Constructor<?> constructor : declaredConstructors) {
    			System.out.println(constructor);
    		}
    
    		System.out.println("----------获取Person类内所有的构造方法,包括私有化构造方法---------------------");
    		System.out.println();
    
    		/*
    		 * 根据指定参数类型获取public修饰的构造方法对象 如果没有指定参数的构造方法,运行异常java.lang.NoSuchMethodException
    		 */
    		Constructor<?> constructor1 = cls.getConstructor();
    //		Constructor<?> constructor2 = cls.getConstructor(int.class);
    		Constructor<?> constructor3 = cls.getConstructor(int.class, String.class);
    		System.out.println(constructor1);
    //		System.out.println(constructor2);
    		System.out.println(constructor3);
    
    // constructor为private修饰的方法,不能获取
    		System.out.println("----------指定参数类型获取public修饰的---------------------------");
    		System.out.println();
    
    		/*
    		 * 通过暴力反射可以获取任意权限修饰符,符合参数要求的构造方法对象
    		 */
    		Constructor<?> declaredConstructor1 = cls.getDeclaredConstructor(String.class);
    		System.out.println(declaredConstructor1);
    
    		System.out.println("------通过暴力反射可以获取任意权限修饰符,符合参数要求的构造方法对象--------------");
    		System.out.println();
    
    		/*
    		 * 通过无参数Constructor对象执行newInstance方法 这里明确是一个Customer类型,可以使用强制类型转换
    		 * 这里使用的是public修饰的构造方法
    		 */
    		Customer c1 = (Customer) constructor1.newInstance();
    		System.out.println(c1);
    		System.out.println(new Customer());
    		System.out.println(constructor3.newInstance(1, "aa@bb.cc"));
    
    		System.out.println("----------无参数Constructor对象执行newInstance方法-------------------");
    		System.out.println();
    
    		// 给予通过暴力反射获取到的非公开权限成员变量,成员方法,构造方法,操作权限
    		// 获取到的:构造方法,实例方法,成员变量.setAccessible
    		// 暴力反射的为所欲为操作
    		declaredConstructor1.setAccessible(true);
    		Customer c2 = (Customer) declaredConstructor1.newInstance("aa@bb.cc");
    		System.out.println(c2);
    
    		
    		System.out.println("---练习-----");
    		
    		Constructor<?> declaredConstructor = cls.getDeclaredConstructor(String.class);
    
    		declaredConstructor.setAccessible(true);
    		Customer customer = (Customer) declaredConstructor.newInstance("hahaha");
    
    		System.out.println(customer);	
    	}
    }
    
    2.3 Method成员方法涉及到的方法
    问题:
    	请问调用执行成员方法时,有哪些需要考虑的内容?
    		调用者
    			类名,对象
    		方法名
    		参数
    	
    	如果需要通过Class对象来获取Method对象,你认为有哪些必要的内容需要考虑?
    		参数
    		方法名
    		权限修饰符
    
    Method[] getMethods();
    	获取类内所有public修饰的成员方法,包括从父类继承而来的public修饰方法。
    
    Method[] getDeclaredMethods();
    	暴力反射
    	获取类内所有成员方法,但是不包括从父类继承而来的方法。
    
    Method getMethod(String methodName, Class... parameterTypes);
    	根据指定的方法名和对应的参数类型,获取对应的public修饰的成员方法
    	methodName: 
    		方法名,指定获取的是哪一个方法
    	parameterTypes:
    		Class用于约束当前使用你的参数数据类型
    		... 不定长参数,方法参数个数,顺序,有参无参问题
    	例如:
    		cls是Class类对象
    		cls.getMethod("setName", String.class);
    		cls.getMethod("getName");		
    
    Method getDeclaredMethod(String methodName, Class... parameterTypes);
    	根据指定的方法名和对应的参数类型,获取对应的成员方法,包括私有化成员方法,但是不
    	包括从父类继承而来的方法
    	methodName: 
    		方法名,指定获取的是哪一个方法
    	parameterTypes:
    		Class用于约束当前使用你的参数数据类型
    		... 不定长参数,方法参数个数,顺序,有参无参问题
    	例如:
    		cls是Class类对象
    		cls.getMethod("setName", String.class);
    		cls.getMethod("getName");		
    
    Object invoke(Object obj, Object... arguments);
    	通过Method类对象调用,执行对应的方法,需要的参数
    	obj : 
    		执行当前方法的执行者
    	arguments:
    		Object... 不定长参数,当前方法执行所需的实际参数,
    
    package reflextion;
    
    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    
    /*
     * Method成员方法涉及到的内容
     */
    public class GetMethodObject {
    	public static void main(String[] args) 
    			throws Exception {
    		/*
    		 * 根据指定的包名.类名,获取对应的Class对象
    		 */
    		Class<?> cls = Class.forName("reflextion.Customer");
    		
    		/*
    		 * 获取类内所有public修饰的成员方法,包括从父类继承而来的方法
    		 */
    		Method[] methods = cls.getMethods();
    		for (Method method : methods) {
    			System.out.println(method);
    		}
    		
    		System.out.println("----获取类内所有public修饰的成员方法,包括从父类继承而来的方法-------------");
    		System.out.println();
    		
    		/*
    		 * 获取类内所有成员方法,包括私有化成员方法,但是不包括父类继承而来的方法
    		 */
    		Method[] declaredMethods = cls.getDeclaredMethods();
    		
    		for (Method method : declaredMethods) {
    			System.out.println(method);
    		}
    		
    		System.out.println("-----获取类内所有成员方法,包括私有化成员方法,但是不包括父类继承而来的方法--------");
    		System.out.println();
    		
    		/*
    		 * 根据指定的方法名和参数类型,获取类内public修饰的成员方法
    		 */
    		Method game1 = cls.getMethod("getId");
    		// setId为私有方法,不能访问
    //		Method game2 = cls.getMethod("setId", int.class);
    		
    		System.out.println(game1);
    		
    		System.out.println("----根据指定的方法名和参数类型,获取类内public修饰的成员方法-----");
    		System.out.println();
    		
    		/*
    		 * 根据指定的方法名和参数类型,获取类内private修饰的成员方法
    		 * 方法在获取反射方法时就已经确定,需要传入适当的参数
    		 */
    		Method declaredMethod1 = cls.getDeclaredMethod("setId",int.class);
    		
    		System.out.println(declaredMethod1);
    
    		System.out.println("-----根据指定的方法名和参数类型,获取类内private修饰的成员方法-------");
    		System.out.println();
    		// 创建实例对象
    		Object obj = cls.getConstructor().newInstance();
    		/*
    		 * 执行public修饰的成员方法
    		 * 依据反射而来的方法对象.invoke(所要执行方法的对象)
    		 */
    		game1.invoke(obj);
    		
    		System.out.println("----执行public修饰的成员方法-------");
    		System.out.println();
    		
    		/*
    		 * 给予暴力反射操作权限的情况下,执行私有化成员方法
    		 */
    		declaredMethod1.setAccessible(true);
    		declaredMethod1.invoke(obj,5);
    		System.out.println(obj);
    		
    		
    		System.out.println("---练习------");
    		Method m1 = cls.getDeclaredMethod("setId",int.class);
    		m1.setAccessible(true);
    		m1.invoke(obj,55);
    		System.out.println(obj);
    	}
    }
    
    2.4 Field成员变量涉及到方法
    Field[] getFields();
    	获取类内所有public修饰的成员变量
    Field[] getDeclaredFields();
    	获取类内所有成员变量,包括私有化成员变量
    
    Field getField(String fieldName);
    	获取指定变量名的成员变量对象,要求是public修饰的成员变量
    
    Field getDeclaredField(String fieldName);
    	获取指定变量名的成员变量对象,包括private私有化修饰的成员变量
    	
    void set(Object obj, Object value);
    	设置指定调用者中对应成员变量的数据
    	obj : 调用者
    	value: 对应当前成员变量需要赋值的内容
    Object get(Object obj);
    	获取指定调用者中指定成员变量的数据
    	obj: 调用者
    
    package reflextion;
    
    
    
    import java.lang.reflect.Field;
    import java.lang.reflect.InvocationTargetException;
    
    /*
     * 获取成员变量Field对象
     */
    public class GetFieldObject {
    	public static void main(String[] args) 
    			throws Exception {
    		/*
    		 * 根据指定的包名.类名,获取对应的Class对象
    		 */
    		Class<?> cls = Class.forName("reflextion.Customer");
    		
    		/*
    		 * 获取本身和父类的public属性
    		 */
    		Field[] fields = cls.getFields();
    		for (Field field : fields) {
    			System.out.println(field);
    		}
    		
    		System.out.println("--------获取对应的public属性-----------");
    		System.out.println();
    		
    		
    		/*
    		 * 获取本身属性
    		 */
    		Field[] declaredFields = cls.getDeclaredFields();
    		for (Field field : declaredFields) {
    			System.out.println(field);
    		}
    		
    		System.out.println("-----获取本身属性------");
    		System.out.println();
    		
    		/*
    		 * 获取public属性id
    		 */
    		Field id = cls.getField("id");
    		System.out.println(id);
    		Field BigName = cls.getField("BigName");
    		System.out.println(BigName);
    		
    		System.out.println("-------获取public属性id------");
    		System.out.println();
    		
    		
    		/*
    		 * 获取本身private属性email
    		 */
    		Field email = cls.getDeclaredField("email");
    		email.setAccessible(true);
    		System.out.println(email);
    		
    		System.out.println("--获取本身private属性email------------");
    		System.out.println();
    		
    		// 创建实例对象
    		Object obj = cls.getConstructor().newInstance();
    		System.out.println(obj+"实例对象");
    		
    		BigName.set(obj, "大名");
    		System.out.println(obj);
    		
    		
    		id.set(obj, 1);
    		email.set(obj, "aa@bb.cc");
    		
    		System.out.println(obj);
    		
    		System.out.println(BigName.get(obj));
    		System.out.println(id.get(obj));
    		System.out.println(email.get(obj));
    		
    		System.out.println("--练习------");
    		Field e = cls.getDeclaredField("email");
    		Object o = cls.getConstructor().newInstance();
    		e.setAccessible(true);
    		e.set(o, "cc.dd@dd.cc");
    		System.out.println(e.get(o));
    		
    	}
    }
    
    2.5 给予暴力反射私有化内容的权限操作
    setAccessible(boolean flag);
    给予Constructor,Method, Field对象,私有化内容,操作权限设置
    true表示可以操作
    
  • 相关阅读:
    screenHeight和windowHeight的区别
    背景图及首次引导加载图片会闪烁、图片预加载
    项目模板
    后端返回一段字符串模板,前端如何解析?
    echarts 饼图label显示指定数据
    html2canvas页面生成图片并添加水印
    js 面试题总结
    如何快速绘制任意角度的扇形?
    纯HTML+CSS 拉窗帘效果
    常用regex正则表达式
  • 原文地址:https://www.cnblogs.com/raising/p/13191440.html
Copyright © 2020-2023  润新知