什么是反射机制
百度百科上的定义是这么说的,
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
JAVA反射(放射)机制:“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言”。从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是动态语言。但是JAVA有着一个非常突出的动态相关机制:Reflection,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods。
介绍反射机制的实现前先说下java的运行机制
java程序经过编译后形成*.class文件,内含JVM的字节码。通过类加载器将字节码(*.class)加载入JVM的内存中。类加载过程主要涉及JVM的方法区。方法区存储了类的类型信息,如运行时常量池(Runtime Constant Pool)、字段和方法数据、构造函数和普通方法的字节码内容、还包括一些在类、实例、接口初始化时用到的特殊方法。这些都可以看作静态信息,每次方法被调用,在java栈中保持该方法的临时变量,当方法返回时,java栈自动撤消。JVM将类加载过程分成加载,连接,初始化三个阶段,其中连接阶段又细分为验证,准备,解析三个阶段。
Java程序开始运行前,类加载器加载类名.class,将类信息保存在运行时数据区的方法区。方法区是线程共享的,里面存储的信息有一个共同的特定就是整个程序中是唯一的。所以有关class的版本号,常量池,方法的字节码等。而类的非静态域则是对象相关的,存储在堆区,其引用(类似指针)存放在栈区
Java程序在运行时,Java运行时系统一直对所有的对象进行所谓的运行时类型标识。这项信息纪录了每个对象所属的类。虚拟机通常使用运行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类。
下面是一段来自网上的代码告诉我们什么是反射
- import java.lang.reflect.Array;
- import java.lang.reflect.Constructor;
- import java.lang.reflect.Field;
- import java.lang.reflect.Method;
- /**
- * Java Reflection Cookbook
- *
- * @author Michael Lee
- * @since 2006-8-23
- * @version 0.1a
- */
- public class Reflection {
- /**
- * 得到某个对象的公共属性
- *
- * @param owner, fieldName
- * @return 该属性对象
- * @throws Exception
- *
- */
- public Object getProperty(Object owner, String fieldName) throws Exception {
- Class ownerClass = owner.getClass();
- Field field = ownerClass.getField(fieldName);
- Object property = field.get(owner);
- return property;
- }
- /**
- * 得到某类的静态公共属性
- *
- * @param className 类名
- * @param fieldName 属性名
- * @return 该属性对象
- * @throws Exception
- */
- public Object getStaticProperty(String className, String fieldName)
- throws Exception {
- Class ownerClass = Class.forName(className);
- Field field = ownerClass.getField(fieldName);
- Object property = field.get(ownerClass);
- return property;
- }
- /**
- * 执行某对象方法
- *
- * @param owner
- * 对象
- * @param methodName
- * 方法名
- * @param args
- * 参数
- * @return 方法返回值
- * @throws Exception
- */
- public Object invokeMethod(Object owner, String methodName, Object[] args)
- throws Exception {
- Class ownerClass = owner.getClass();
- Class[] argsClass = new Class[args.length];
- for (int i = 0, j = args.length; i < j; i++) {
- argsClass[i] = args[i].getClass();
- }
- Method method = ownerClass.getMethod(methodName, argsClass);
- return method.invoke(owner, args);
- }
- /**
- * 执行某类的静态方法
- *
- * @param className
- * 类名
- * @param methodName
- * 方法名
- * @param args
- * 参数数组
- * @return 执行方法返回的结果
- * @throws Exception
- */
- public Object invokeStaticMethod(String className, String methodName,
- Object[] args) throws Exception {
- Class ownerClass = Class.forName(className);
- Class[] argsClass = new Class[args.length];
- for (int i = 0, j = args.length; i < j; i++) {
- argsClass[i] = args[i].getClass();
- }
- Method method = ownerClass.getMethod(methodName, argsClass);
- return method.invoke(null, args);
- }
- /**
- * 新建实例
- *
- * @param className
- * 类名
- * @param args
- * 构造函数的参数
- * @return 新建的实例
- * @throws Exception
- */
- public Object newInstance(String className, Object[] args) throws Exception {
- Class newoneClass = Class.forName(className);
- Class[] argsClass = new Class[args.length];
- for (int i = 0, j = args.length; i < j; i++) {
- argsClass[i] = args[i].getClass();
- }
- Constructor cons = newoneClass.getConstructor(argsClass);
- return cons.newInstance(args);
- }
- /**
- * 是不是某个类的实例
- * @param obj 实例
- * @param cls 类
- * @return 如果 obj 是此类的实例,则返回 true
- */
- public boolean isInstance(Object obj, Class cls) {
- return cls.isInstance(obj);
- }
- /**
- * 得到数组中的某个元素
- * @param array 数组
- * @param index 索引
- * @return 返回指定数组对象中索引组件的值
- */
- public Object getByArray(Object array, int index) {
- return Array.get(array,index);
- }
- }