Java反射机制

Java反射机制

 

什么是反射?

Java反射机制是指在运行状态中,对于任意一个类,都能获得这个类的所有属性和方法,对任意的一个属性和方法都能调用他的任意一个属性和方法。这种在运行时动态的获取信息以及动态调用对象的方法的功能称为Java的反射机制。

反射机制的实质是将类中的各个成分映射成一个个对象,比如一个类具有的的属性,方法,构造方法都可以利用反射机制将一个个成分剖析成为一个个对象,因为某种程度上来说属性,方法,构造方法也是一个对象

Java将与反射有关的类都封装在了import java.lang.reflect包下

如何建立反射

建立反射的第一步就是获取类的Class对象,有三种方法可以获取Class对象

1.用Object.getClass()方法获取Class对象

2.用Class.forName()方法获取

3.用类名.Class方法获取

然后可以通过Class对象获取这个类的属性,方法,构造方法

可以利用构造方法实例化对象,利用Method.invoke(obj)方法调用方法

在这里需要注意的是如果这个属性,方法,构造方法是private,protect修饰的,则在获取到该对象之后需要使用其父类的setAccessible(true)方法,将标志位设为true(取消了Java的权限控制检查),否则会报java.lang.IllegalAccessException错

但是这并不意味着public和默认的修饰的属性方法的标志位为true,也就是说Accessible标志位并不是标志方法,属性能否访问的

但是如果使用该方法会大大提升效率(大约20倍的速度)

来自API文档中的解释:

引用
AccessibleObject 类是 Field、Method 和 Constructor 对象的基类。它提供了将反射的对象标记为在使用时取消默认 Java 语言访问控制检查的能力。对于公共成员、默认(打包)访问成员、受保护成员和私有成员,在分别使用 Field、Method 或 Constructor 对象来设置或获得字段、调用方法,或者创建和初始化类的新实例的时候,会执行访问检查。
在反射对象中设置 accessible 标志允许具有足够特权的复杂应用程序(比如 Java Object Serialization 或其他持久性机制)以某种通常禁止使用的方式来操作对象。
setAccessible 
public void setAccessible(boolean flag) 
                   throws SecurityException 
将此对象的 accessible 标志设置为指示的布尔值。值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。值为 false 则指示反射的对象应该实施 Java 语言访问检查。
实际上setAccessible是启用和禁用访问安全检查的开关,并不是为true就能访问为false就不能访问

下面放上我测试的代码:

public void create(){
		
		try {
			//获取Class对象
			Class s=Class.forName("reflect.A");
			
			//获取A的构造方法
			Constructor tor=s.getConstructor();//访问无参的构造方法
			Constructor con=s.getConstructor(String.class);//访问公有的构造方法
			
			Constructor pre=s.getDeclaredConstructor(int.class);//访问私有的构造方法
			pre.setAccessible(true);
			//创建对象
			Object ob=s.newInstance();//s.getConstructor(String.class,int.class);
			
			Object j=tor.newInstance();//根据无参构造方法创建对象
			Object o= con.newInstance("data");//根据公有构造方法创建对象
			
			Object b=pre.newInstance(11);//根据私有构造方法创建对象
			//获取类的方法
			Method m= s.getDeclaredMethod("gets");
			
			Method e=s.getMethod("geta" );
			m.setAccessible(true);
			//获取类的属性
			Field i=s.getField("s");
			Field f=s.getDeclaredField("a");
			f.setAccessible(true);
			//调用方法
			m.invoke(o);
			e.invoke(b);
			System.out.println(i.get(j));
			System.out.println(f.get(o));
			
			//m.invoke(ob, "something",99);
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		} catch (NoSuchMethodException e) {
			e.printStackTrace();
		} catch (SecurityException e) {
			e.printStackTrace(); 
		} catch (InstantiationException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			e.printStackTrace();
		} catch (NoSuchFieldException e1) {
			e1.printStackTrace();
		}
		
	}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值