Java反射机制

Java反射机制

 

Class反射对象描述类语义结构,可以从Class对象中获取构造函数、成员变量、方法类等类元素的反射对象,并以编程的方式通过这些反射对象对目标对象进行操作。

 

这些反射对象类在java.reflect包中定义,下面是最主要的三个反射类:

 

Constructor:类的构造函数反射类,通过Class#getConstructors()方法可以获得类的所有构造函数反射对象数组。在JDK5.0中,还可以通过getConstructor(Class... parameterTypes)获得拥有特定入参的构造函数反射对象。Constructor的一个主要方法是newInstance(Object[ ] initargs),通过该方法可以创建一个对象类的实例,相当于new关键字。在JDK5.0中该方法演化为更为灵活的形式:newInstance(Object... initargs)。

 

Method:类方法的反射类,通过Class#getDeclaredMethods()方法可以获得类的所有方法反射类对象数组

Method[ ]。在JDK5.0中可以通过getDeclaredMethod(String name,Class... parameterTypes)获取特定签名的方法,name为方法名;Class...为方法入参类型列表。

 

Method最主要的方法是invoke(Object obj,Object[] args),obj表示操作的目标对象;args为方法入参,在JDK5.0中,该方法的形式调整为invoke(Object obj,Object...args)。

 

此外,Method还有很多用于获取类方法更多信息的方法:

 

1)Class getReturnType(): 获取方法的返回值类型

2)Class[ ] getParameterTypes(): 获取方法的入参类型数组

3)Class[ ] getExceptionTypes(): 获取方法的异常类型数组

4)Annotation[ ][ ] getParameterAnnotations(): 获取方法的注解信息,JDK5.0中的新方法

 

Field:类的成员变量的反射类,通过Class#getDeclaredFields()方法可以获取类的成员变量反射对象数组,通过Class#getDeclaredField(String name)则可获去某个特定名称的成员变量反射对象。

 

Field类最主要的方法是set(Object obj,Object value),obj表示操作的目标对象, 通过value对目标对象的成员变量设置值。如果成员变量为基础类型,用户可以使用Field类中提供的带类型名的值设置方法,如setBoolean(Object obj,boolean value)、setInt(Object obj,int value)等。

 

此外,Java还为包提供了Package反射类,在JDK5.0中还为注解提供了AnnotatedElement反射类。总之,Java的反射体系保证了可以通过程序化的方式访问目标类中所有的元素,对于private或protected的成员变量和方法,只要JVM的安全机制允许,也可以通过反射进行调用,请看下面的例子:

 

 

package com.fortune.test.reflect;

/**
 * Created with IntelliJ IDEA.
 * User: Alan
 * Date: 12-5-17
 * Time: 下午1:44
 */
public class PrivateCar {
    private String color;

    protected void drive(){
        System.out.println("drive private car!the color is:"+color);
    }
}
 

 

 

color变量和drive()方法都是私有的,通过类实例变量无法在外部访问私有变量、调用私有方法的,但通过反射机制却可以绕过这个限制:

 

 

package com.fortune.test.reflect;

import java.lang.reflect.Field;
import java.lang.reflect.Method;

/**
 * Created with IntelliJ IDEA.
 * User: Alan
 * Date: 12-5-17
 * Time: 下午1:52
 */
public class PrivateCarReflect {
    public static void main(String[] args) throws Throwable {
        ClassLoader loader = Thread.currentThread().getContextClassLoader();
        Class clazz = loader.loadClass("com.fortune.test.reflect.PrivateCar");
        PrivateCar pcar = (PrivateCar) clazz.newInstance();

        Field colorFld = clazz.getDeclaredField("color");
        //①取消Java语言访问检查以访问private变量
        colorFld.setAccessible(true);
        colorFld.set(pcar, "红色");

        Method driveMtd = clazz.getDeclaredMethod("drive",(Class[])null);
        //Method driveMtd = clazz.getDeclaredMethod("drive"); JDK5.0下使用

        //②取消Java语言访问检查以访问protected方法
        driveMtd.setAccessible(true);
        driveMtd.invoke(pcar, (Object[]) null);

    }
}

 

 

运行该类,打印出以下信息:

 

drive private car! the color is: 红色

 

在访问private、protected成员变量和方法时必须通过setAccessible(boolean access)方法取消Java语言检查,否则将抛出IllegalAccessException。如果JVM的安全管理器设置了相应的安全机制,调用该方法将抛出SecurityException。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值