JJJJavaavaavaava反射机制文章集合关键字:JavaJavaJavaJava反射ClassClassClassClass由“彭益存”从网络整理Email:Email:Email:Email:yicun05@software.nju.edu.cnyicun05@software.nju.edu.cnyicun05@software.nju.edu.cnyicun05@software.nju.edu.cnpengyicun05@163.compengyicun05@163.compengyicun05@163.compengyicun05@163.com2009/03/252009/03/252009/03/252009/03/25目录1.Java反射机制2.谈java反射机制3.Java反射机制4.Java集合浅析5.Java的类反射机制6.Java反射机制的学习7.Java反射Reflection--运行时生成实例8.Java反射机制9.Java反射机制210.Java反射机制(转)注意:所有文章均从网络途径获得,本人只作收集整理的工作,个别地方有改动。每篇文章的标题遵循原作者的表述,本人不再作任何更改。Java反射机制反射的概念反射的概念反射的概念反射的概念::::反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。其中LEAD/LEAD++、OpenC++、MetaXa和OpenJava等就是基于反射机制的语言。最近,反射机制也被应用到了视窗系统、操作系统和文件系统中。反射本身并不是一个新概念,它可能会使我们联想到光学中的反射概念,尽管计算机科学赋予了反射概念新的含义,但是,从现象上来说,它们确实有某些相通之处,这些有助于我们的理解。在计算机科学领域,反射是指一类应用,它们能够自描述和自控制。也就是说,这类应用通过采用某种机制来实现对自己行为的描述(self-representation)和监测(examination),并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。可以看出,同一般的反射概念相比,计算机科学领域的反射不单单指反射本身,还包括对反射结果所采取的措施。所有采用反射机制的系统(即反射系统)都希望使系统的实现更开放。可以说,实现了反射机制的系统都具有开放性,但具有开放性的系统并不一定采用了反射机制,开放性是反射系统的必要条件。一般来说,反射系统除了满足开放性条件外还必须满足原因连接(Causally-connected)。所谓原因连接是指对反射系统自描述的改变能够立即反映到系统底层的实际状态和行为上的情况,反之亦然。开放性和原因连接是反射系统的两大基本要素。13700863760Java中,反射是一种强大的工具。它使您能够创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代表链接。反射允许我们在编写与执行时,使我们的程序代码能够接入装载到JVM中的类的内部信息,而不是源代码中选定的类协作的代码。这使反射成为构建灵活的应用的主要工具。但需注意的是:如果使用不当,反射的成本很高。JavaJavaJavaJava中的类反射:中的类反射:中的类反射:中的类反射:Reflection是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查,或者说“自审”,并能直接操作程序的内部属性。Java的这一能力在实际应用中也许用得不是很多,但是在其它的程序设计语言中根本就不存在这一特性。例如,Pascal、C或者C++中就没有办法在程序中获得函数定义相关的信息。1111.检测类:1.11.11.11.1reflectionreflectionreflectionreflection的工作机制的工作机制的工作机制的工作机制考虑下面这个简单的例子,让我们看看reflection是如何工作的。importjava.lang.reflect.*;publicclassDumpMethods{publicstaticvoidmain(Stringargs[]){try{Classc=Class.forName(args[0]);Methodm[]=c.getDeclaredMethods();for(inti=0;im.length;i++)System.out.println(m[i].toString());}catch(Throwablee){System.err.println(e);}}}按如下语句执行:javaDumpMethodsjava.util.Stack它的结果输出为:publicjava.lang.Objectjava.util.Stack.push(java.lang.Object)publicsynchronizedjava.lang.Objectjava.util.Stack.pop()publicsynchronizedjava.lang.Objectjava.util.Stack.peek()publicbooleanjava.util.Stack.empty()publicsynchronizedintjava.util.Stack.search(java.lang.Object)这样就列出了java.util.Stack类的各方法名以及它们的限制符和返回类型。这个程序使用Class.forName载入指定的类,然后调用getDeclaredMethods来获取这个类中定义了的方法列表。java.lang.reflect.Methods是用来描述某个类中单个方法的一个类。1.21.21.21.2JavaJavaJavaJava类反射中的主要方法类反射中的主要方法类反射中的主要方法类反射中的主要方法对于以下三类组件中的任何一类来说--构造函数、字段和方法--java.lang.Class提供四种独立的反射调用,以不同的方式来获得信息。调用都遵循一种标准格式。以下是用于查找构造函数的一组反射调用:lConstructorgetConstructor(Class[]params)--获得使用特殊的参数类型的公共构造函数,lConstructor[]getConstructors()--获得类的所有公共构造函数lConstructorgetDeclaredConstructor(Class[]params)--获得使用特定参数类型的构造函数(与接入级别无关)lConstructor[]getDeclaredConstructors()--获得类的所有构造函数(与接入级别无关)获得字段信息的Class反射调用不同于那些用于接入构造函数的调用,在参数类型数组中使用了字段名:lFieldgetField(Stringname)--获得命名的公共字段lField[]getFields()--获得类的所有公共字段lFieldgetDeclaredField(Stringname)--获得类声明的命名的字段lField[]getDeclaredFields()--获得类声明的所有字段用于获得方法信息函数:lMethodgetMethod(Stringname,Class[]params)--使用特定的参数类型,获得命名的公共方法lMethod[]getMethods()--获得类的所有公共方法lMethodgetDeclaredMethod(Stringname,Class[]params)--使用特写的参数类型,获得类声明的命名的方法lMethod[]getDeclaredMethods()--获得类声明的所有方法1.31.31.31.3开始使用开始使用开始使用开始使用ReflectionReflectionReflectionReflection::::用于reflection的类,如Method,可以在java.lang.relfect包中找到。使用这些类的时候必须要遵循三个步骤:第一步是获得你想操作的类的java.lang.Class对象。在运行中的Java程序中,用java.lang.Class类来描述类和接口等。下面就是获得一个Class对象的方法之一:Classc=Class.forName(java.lang.String);这条语句得到一个String类的类对象。还有另一种方法,如下面的语句:Classc=int.class;或者Classc=Integer.TYPE;它们可获得基本类型的类信息。其中后一种方法中访问的是基本类型的封装类(如Integer)中预先定义好的TYPE字段。第二步是调用诸如getDeclaredMethods的方法,以取得该类中定义的所有方法的列表。一旦取得这个信息,就可以进行第三步了——使用reflectionAPI来操作这些信息,如下面这段代码:Classc=Class.forName(java.lang.String);Methodm[]=c.getDeclaredMethods();System.out.println(m[0].toString());它将以文本方式打印出String中定义的第一个方法的原型。2222.处理对象:如果要作一个开发工具像debugger之类的,你必须能发现filedvalues,以下是三个步骤:a.创建一个Class对象b.通过getField创建一个Field对象c.调用Field.getXXX(Object)方法(XXX是Int,Float等,如果是对象就省略;Object是指实例).例如:importjava.lang.reflect.*;importjava.awt.*;classSampleGet{publicstaticvoidmain(String[]args){Rectangler=newRectangle(100,325);printHeight(r);}staticvoidprintHeight(Rectangler){FieldheightField;IntegerheightValue;Classc=r.getClass();try{heightField=c.getField(height);heightValue=(Integer)heightField.get(r);System.out.println(Height:+heightValue.toString());}catch(NoSuchFieldExceptione){System.out.println(e);}catch(SecurityExceptione){System.out.println(e);}catch(IllegalAccessExceptione){System.out.println(e);}}}安全性和反射:安全性和反射:安全性和反射:安全性和反射:在处理反射时安全性是一个较复杂的问题。反射经常由框架型代码使用,由于这一点,我们可能希望框架能够全面接入代码,无需考虑常规的接入限制。但是,在其它情况下,不受控制的接入会带来严重的安全性风险,例如当代码在不值得信任的代码共享的环境中运行时。由于这些互相矛盾的需求,Java编程语言定义一种多级别方法来处理反射的安全性。基本模式是对反射实施与应用于源代码接入相同的限制:n从任意位置到类公共组件的接入n类自身外部无任何到私有组件的接入n受保护和打包(缺省接入)组件的有限接入不过至少有些时候,围绕这些限制还有一种简单的方法。我们可以在我们所写的类中,扩展一个普通的基本类java.lang.reflect.AccessibleObject类。这个类定义了一种setAccessible方法,使我们能够启动或关闭对这些类中其中一个类的实例的接入检测。唯一的问题在于如果使用了安全性管理器,它将检测正在关闭接入检测的代码是否许可了这