Java中的反射

反射
说起来反射,我们需要明确Java文件和.class文件的关系
笼统的说:java文件通过编译生成.class文件这就是他们之间的关系
细致的说:java文件中含代码的所有内容:成员变量,构造方法,成员方法…编译成对应的
.class文件,我们将编译好的.class文件放进内存的代码区,.class字节码文件所占的内存空间就可以看作一个Class类对象:构造方法Constructor,成员方法Method,成员变量Filed,注解Annotation把以上的四个内容,看作一个Class类对象的属性,提供操作的方法和执行方式。通过反射的方式可以让代码又更好的普适性
.class字节码问题:
.java文件,通过javac编译工具生成对应的.class字节码文件
使用JDK提供的反编译工具,可以看到.class文件中包含
class 完整的包名.类名
Filed 成员变量,成员变量的名字和成员变量的数据类型【如果是引用数据类型,也是完整的包名.类名】
Method 成员方法,方法权限修饰符,返回值类型,方法名,形参列表数据类型(如果是引用数据数据类型,就是完整的包名.类名)
代码区存在一块空间 ==> .class ==> Java程序的所有内容
在这里插入图片描述

反射中的方法

1.Class涉及到的方法
Class Class.forName(String packageNameAndClassName);
Class类的静态成员方法,通过完整的包名.类名获取对应.class文件的Class对象,同事也可以作为.class文件加载的方法。
Class 类名.class;
通过类名.class方法,获取对应.class文件的Class对象,通常用于方法的参数类型。
Class 类对象.getClass();
通过类对象.getClass获得.class文件对应的Class对象,或者说数据类型判断

package com.qfedu.a_reflect;

public class Demo1 {
	public static void main(String[] args) throws ClassNotFoundException {
		Class<?> forName = Class.forName("com.qfedu.a_reflect.Person");
		Class<?> forName1 = Class.forName("com.qfedu.a_reflect.Person");
		  Class<com.qfedu.a_reflect.Person> cls = Person.class;
		  Class<? extends Person> class1 = new Person().getClass();
		  System.out.println(forName == cls);
		  System.out.println(forName == forName1);
		  System.out.println(forName == class1);
		  System.out.println(cls == class1);
		  
	}
}

** 2.Constructor 构造方法类涉及到的方法**
public Constructor[] getConstructors();
当前Class类对象对应的Java文件中所有public修饰的构造方法的数组
public Constructor[] getDeclaredConstructors();
【暴力反射】
当前Class类对象对应的Java文件中所有构造方法的数组,包括私有化的构造方法
public Constructor getConstructor(Class… initArgumentTypes);
根据指定的数据类型,来选择对应的构造方法,这里可能抛出异常NoSuchMethodException
这里有且只能获取获取类内的指定数据类型public修饰构造方法类对象
Class: 约束数据类型,当前方法所需的参数类型
例如:
这里需要int类型 int.class
这里需要String类型 String.class
之类需要Perosn类型 Person.class
… : 不定长参数
构造方法需要的参数类型是很多的,有可能无参数,有可能有参数。… 不定长参数
类约束使用,增强代码的普适性
initArgumentTypes:
参数名 初始化参数类型复数
public Constructor getDeclaredConstructor(Class…);
获取当前Class类对象对应的java文件中是指定数据类型的构造方法可以是私有的
【暴力反射】
这里有且只能获取获取类内的指定数据类型public修饰构造方法类对象
Class: 约束数据类型,当前方法所需的参数类型
例如:
这里需要int类型 int.class
这里需要String类型 String.class
之类需要Perosn类型 Person.class
… : 不定长参数
构造方法需要的参数类型是很多的,有可能无参数,有可能有参数。… 不定长参数
类约束使用,增强代码的普适性
initArgumentTypes:
参数名 初始化参数类型复数
Object newInstance(Object… initArguments);
通过Constructor对象来调用,传入当前构造方法所需创建对象的初始化参数,创建对象。
Object: Object类是Java中所有类的基类,这里可以传入任意类型的参数
… : 不定长参数,因为Constructor类对象在获取的过程中,约束的参数个数都不确定,
这里使用不定长参数来传入数据

package com.qfedu.a_reflect;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Demo2 {
		public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
			Class<?> cls = Class.forName("com.qfedu.a_reflect.Person");
			Constructor<?>[] constructors = cls.getConstructors();
			System.out.println("当前Class类对象对应的Java文件中public修饰的构造方法");
			for (Constructor<?> constructor : constructors) {
				System.out.println(constructor);
			}
			Constructor<?>[] declaredConstructors = cls.getDeclaredConstructors();
			System.out.println("当前Class类对象对应的Java文件中所有的构造方法");
			for (Constructor<?> constructor : declaredConstructors) {
				System.out.println(constructor);
			}
			System.out.println("--------------------------------------------------------");
			Constructor<?> constructor = cls.getConstructor();
			Constructor<?> constructor2 = cls.getConstructor(int.class);
			Constructor<?> constructor3 = cls.getConstructor(int.class,String.class);
			//Constructor<?> constructor4 = cls.getConstructor(String.class);
			//System.out.println(constructor4);
			System.out.println(constructor);
			System.out.println(constructor2);
			System.out.println(constructor3);
			Constructor<?> declaredConstructor = cls.getDeclaredConstructor(String.class);
			System.out.println(declaredConstructor);
			Constructor<?> declaredConstructor3 = cls.getDeclaredConstructor(int.class,String.class);
			Constructor<?> declaredConstructor2 = cls.getDeclaredConstructor();
			System.out.println(declaredConstructor2);
			System.out.println("--------------------------------------------------------");
			//Object obj = cls.getConstructor().newInstance();
			declaredConstructor3.setAccessible(true);
			declaredConstructor.setAccessible(true);
//用到的暴力反射都必须事先进行私有化权限的操作
			Object obj1 = declaredConstructor3.newInstance(18,"鲁千旗");
			Object obj = declaredConstructor.newInstance("鲁千旗");
			System.out.println(obj1);
		}
}

Method成员方法涉及到的方法
public Method[] getMethods();
获得Class类对象对应的java文件中public修饰的成员方法组成的数组,包括从父类继承而来的public修饰方法。
public Method[] getDeclaredMethods();
获得Class类对象对应的Java文件中所有的成员方法组成的数组,但是不包括从父类继承而来的方法。
public Method getMethod(String MethodName,Class…parameterTypes);
根据指定方法名和数据类型,获取对应的public修饰的成员方法
methodName:
方法名,指定获取的是哪一个方法
parameterTypes:
Class用于约束当前使用你的参数数据类型
… 不定长参数,方法参数个数,顺序,有参无参问题
例如:
cls是Class类对象
cls.getMethod(“setName”, String.class);
cls.getMethod(“getName”);
public 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 com.qfedu.a_reflect;

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

public class Demo3 {
		public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
			Class<?> cls = Class.forName("com.qfedu.a_reflect.Person");
			Method[] methods = cls.getMethods();
			for (Method method : methods) {
				System.out.println(method);
			}
			System.out.println("----------------------------------");
			Method[] declaredMethods = cls.getDeclaredMethods();
			for (Method method : declaredMethods) {
				System.out.println(method);
			}
			System.out.println("----------------------------------");
			
			
			Method method = cls.getMethod("game");
			Method method2 = cls.getMethod("game", String.class);
			Object obj = cls.getConstructor().newInstance();
			//引用,获得的方法需要类对象引用
			 method.invoke(obj);
			 method2.invoke(obj, "英雄联盟");
			 Method declaredMethod = cls.getDeclaredMethod("testPrivate");
			 Method declaredMethod2 = cls.getDeclaredMethod("testPrivate", String.class);
			 declaredMethod2.setAccessible(true);
			 declaredMethod.setAccessible(true);
			 declaredMethod2.invoke(obj, "粑粑");
			 declaredMethod.invoke(obj);
			 
		}
}

Field成员变量涉及到方法
public Filed[] getFileds();
获取当前Class对象对应的java文件中,所有public修饰的成员变量数组
public Filed[] getDeclaredFileds();
获取当前Class对象对应的java文件中,所有成员变量数组,包括私有成员变量
public Filed getFiled(String filedName);
获取指定变量名的成员变量对象,要求是public修饰的成员变量
public Filed getDeclaredFiled(String name);
获取指定变量名的成员变量对象,包括private私有化修饰的成员变量
void set(Object obj, Object value);
设置指定调用者中对应成员变量的数据
obj : 调用者
value: 对应当前成员变量需要赋值的内容
Object get(Object obj);
获取指定调用者中指定成员变量的数据
obj: 调用者

package com.qfedu.a_reflect;

import java.io.File;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;

public class Demo4 {
	public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException, NoSuchFieldException {
		Class<?> cls = Class.forName("com.qfedu.a_reflect.Person");
		Constructor<?> declaredConstructor = cls.getDeclaredConstructor(int.class,String.class);
		declaredConstructor.setAccessible(true);
		Object obj = declaredConstructor.newInstance(22,"lqq");
		Field field = cls.getDeclaredField("test");
		Field field1 = cls.getDeclaredField("id");
		Field field2 = cls.getDeclaredField("name");
		field.setAccessible(true);
		field1.setAccessible(true);
		field2.setAccessible(true);
		field.set(obj, 19999);
		System.out.println(obj);
		Object test = field.get(obj);
		Object id = field1.get(obj);
		Object name = field2.get(obj);
		System.out.println(id);
		System.out.println(name);
		System.out.println(test);
		
	}
}

给予暴力反射私有化内容的权限操作

setAccessible(boolean flag);
给予Constructor,Method, Field对象,私有化内容,操作权限设置
true表示可以操作

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值