Java运行原理和反射机制Word文件下载.docx
《Java运行原理和反射机制Word文件下载.docx》由会员分享,可在线阅读,更多相关《Java运行原理和反射机制Word文件下载.docx(16页珍藏版)》请在冰豆网上搜索。
这样,未知的对象的类信息在运行时就能被完全确定下来,而在编译时不需要知道任何信息。
另外,RTTI有时能解决效率问题。
当程序中使用多态给程序的运行带来负担的时候,可以使用RTTI编写一段代码来提高效率。
Java的反射机制的实现要借助于4个类:
class,Constructor,Field,Method;
其中class代表的时类对象,Constructor-类的构造器对象,Field-类的属性对象,Method-类的方法对象。
通过这四个对象我们可以粗略的看到一个类的各个组成部分。
Class:
程序运行时,java运行时系统会对所有的对象进行运行时类型的处理。
这项信息记录了每个对象所属的类,虚拟机通常使用运行时类型信息选择正确的方法来执行(摘自:
白皮书)。
但是这些信息我们怎么得到啊,就要借助于class类对象了啊。
在Object类中定义了getClass()方法。
我们可以通过这个方法获得指定对象的类对象。
然后我们通过分析这个对象就可以得到我们要的信息了。
比如:
ArrayListarrayList;
Classclazz=arrayList.getClass();
然后我来处理这个对象clazz。
当然了Class类具有很多的方法,这里重点将和Constructor,Field,Method类有关系的方法。
Reflection是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查,或者说“自审”,并能直接操作程序的内部属性。
Java的这一能力在实际应用中也许用得不是很多,但是个人认为要想对java有个更加深入的了解还是应该掌握的。
1.检测类:
reflection的工作机制
考虑下面这个简单的例子,让我们看看reflection是如何工作的。
importjava.lang.reflect.*;
publicclassDumpMethods{
publicstaticvoidmain(Stringargs[]){
try{
Classc=Class.forName(args[0]);
Methodm[]=c.getDeclaredMethods();
for(inti=0;
i<
m.length;
i++)
System.out.println(m[i].toString());
}catch(Throwablee){
System.err.println(e);
}
按如下语句执行:
javaDumpMethodsjava.util.ArrayList
这个程序使用Class.forName载入指定的类,然后调用getDeclaredMethods来获取这个类中定义了的方法列表。
java.lang.reflect.Methods是用来描述某个类中单个方法的一个类。
Java类反射中的主要方法
对于以下三类组件中的任何一类来说--构造函数、字段和方法--java.lang.Class提供四种独立的反射调用,以不同的方式来获得信息。
调用都遵循一种标准格式。
以下是用于查找构造函数的一组反射调用:
ConstructorgetConstructor(Class[]params)--获得使用特殊的参数类型的公共构造函数,
Constructor[]getConstructors()--获得类的所有公共构造函数
ConstructorgetDeclaredConstructor(Class[]params)--获得使用特定参数类型的构造函数(与接入级别无关)
Constructor[]getDeclaredConstructors()--获得类的所有构造函数(与接入级别无关)
获得字段信息的Class反射调用不同于那些用于接入构造函数的调用,在参数类型数组中使用了字段名:
FieldgetField(Stringname)--获得命名的公共字段
Field[]getFields()--获得类的所有公共字段
FieldgetDeclaredField(Stringname)--获得类声明的命名的字段
Field[]getDeclaredFields()--获得类声明的所有字段
用于获得方法信息函数:
MethodgetMethod(Stringname,Class[]params)--使用特定的参数类型,获得命名的公共方法
Method[]getMethods()--获得类的所有公共方法
MethodgetDeclaredMethod(Stringname,Class[]params)--使用特写的参数类型,获得类声明的命名的方法
Method[]getDeclaredMethods()--获得类声明的所有方法
使用Reflection:
用于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;
它们可获得基本类型的类信息。
其中后一种方法中访问的是基本类型的封装类(如Intege)中预先定义好的TYPE字段。
第二步是调用诸如getDeclaredMethods的方法,以取得该类中定义的所有方法的列表。
一旦取得这个信息,就可以进行第三步了——使用reflectionAPI来操作这些信息,如下面这段代码:
System.out.println(m[0].toString());
它将以文本方式打印出String中定义的第一个方法的原型。
处理对象:
a.创建一个Class对象
b.通过getField创建一个Field对象
c.调用Field.getXXX(Object)方法(XXX是Int,Float等,如果是对象就省略;
Object是指实例).
例如:
importjava.awt.*;
classSampleGet{
publicstaticvoidmain(String[]args){
Rectangler=newRectangle(100,325);
printHeight(r);
staticvoidprintHeight(Rectangler){
FieldheightField;
IntegerheightValue;
Classc=r.getClass();
heightField=c.getField("
height"
heightValue=(Integer)heightField.get(r);
System.out.println("
Height:
"
+heightValue.toString());
}catch(NoSuchFieldExceptione){
System.out.println(e);
}catch(SecurityExceptione){
}catch(IllegalAccessExceptione){
安全性和反射:
在处理反射时安全性是一个较复杂的问题。
反射经常由框架型代码使用,由于这一点,我们可能希望框架能够全面接入代码,无需考虑常规的接入限制。
但是,在其它情况下,不受控制的接入会带来严重的安全性风险,例如当代码在不值得信任的代码共享的环境中运行时。
由于这些互相矛盾的需求,Java编程语言定义一种多级别方法来处理反射的安全性。
基本模式是对反射实施与应用于源代码接入相同的限制:
从任意位置到类公共组件的接入
类自身外部无任何到私有组件的接入
受保护和打包(缺省接入)组件的有限接入
不过至少有些时候,围绕这些限制还有一种简单的方法。
我们可以在我们所写的类中,扩展一个普通的基本类java.lang.reflect.AccessibleObject类。
这个类定义了一种setAccessible方法,使我们能够启动或关闭对这些类中其中一个类的实例的接入检测。
唯一的问题在于如果使用了安全性管理器,它将检测正在关闭接入检测的代码是否许可了这样做。
如果未许可,安全性管理器抛出一个例外。
下面是一段程序,在TwoString类的一个实例上使用反射来显示安全性正在运行:
publicclassReflectSecurity{
TwoStringts=newTwoString("
a"
"
b"
Fieldfield=clas.getDeclaredField("
m_s1"
//field.setAccessible(true);
Retrievedvalueis"
+
field.get(inst));
}catch(Exceptionex){
ex.printStackTrace(System.out);
如果我们编译这一程序时,不使用任何特定参数直接从命令行运行,它将在field.get(inst)调用中抛出一个IllegalAccessException异常。
如果我们不注释field.setAccessible(true)代码行,那么重新编译并重新运行该代码,它将编译成功。
最后,如果我们在命令行添加了JVM参数-Djava.security.manager以实现安全性管理器,它仍然将不能通过编译,除非我们定义了ReflectSecurity类的许可权限。
反射性能:
(转录别人的啊)
反射是一种强大的工具,但也存在一些不足。
一个主要的缺点是对性能有影响。
使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它满足我们的要求。
这类操作总是慢于只直接执行相同的操作。
下面的程序是字段接入性能测试的一个例子,包括基本的测试方法。
每种方法测试字段接入的一种形式--accessSame与同一对象的成员字段协作,accessOther使用可直接接入的另一对象的字段,accessReflection使用可通过反射接入的另一对象的字段。
在每种情况下,方法执行相同的计算--循环中简单的加/乘顺序。
程序如下:
publicintaccessSame(intloops){
m_value=0;
for(intindex=0;
index<
loops;
index++){
m_value=(m_value+ADDITIVE_VALUE)*
MULTIPLIER_VALUE;
returnm_value;
publicintacces
sReference(intloops){
TimingClasstiming=newTimingClass();
timing.m_value=(timing.m_value+ADDITIVE_VALUE)*
returntiming.m_value;
publicintaccessReflection(intloops)throwsException{
Fieldfield=TimingClass.class.
getDeclaredField("
m_value"
intvalue=(field.getInt(timing)+
ADDITIVE_VALUE)*MULTIPLIER_VALUE;
field.setInt(timing,value);
Errorusingreflection"
throwex;
在上面的例子中,测试程序重复调用每种方法,使用一个大循环数,从而平均多次调用的时间衡量结果。
平均值中不包括每种方法第一次调用的时间,因此初始化时间不是结果中的一个因素。
下面的图清楚的向我们展示了每种方法字段接入的时间:
图1:
字段接入时间:
我们可以看出:
在前两副图中(SunJVM),使用反射的执行时间超过使用直接接入的1000倍以上。
通过比较,IBMJVM可能稍好一些,但反射方法仍旧需要比其它方法长700倍以上的时间。
任何JVM上其它两种方法之间时间方面无任何显著差异,但IBMJVM几乎比SunJVM快一倍。
最有可能的是这种差异反映了SunHotSpotJVM的专业优化,它在简单基准方面表现得很糟糕。
反射性能是Sun开发1.4JVM时关注的一个方面,它在反射方法调用结果中显示。
在这类操作的性能方面,Sun1.4.1JVM显示了比1.3.1版本很大的改进。
如果为为创建使用反射的对象编写了类似的计时测试程序,我们会发现这种情况下的差异不象字段和方法调用情况下那么显著。
使用newInstance()调用创建一个简单的java.lang.Object实例耗用的时间大约是在Sun1.3.1JVM上使用newObject()的12倍,是在IBM1.4.0JVM的四倍,只是Sun1.4.1JVM上的两部。
使用Array.newInstance(type,size)创建一个数组耗用的时间是任何测试的JVM上使用newtype[size]的两倍,随着数组大小的增加,差异逐步缩小。
随着jdk6.0的推出,反射机制的性能也有了很大的提升。
期待中….
总结:
Java语言反射提供一种动态链接程序组件的多功能方法。
它允许程序创建和控制任何类的对象(根据安全性限制),无需提前硬编码目标类。
这些特性使得反射特别适用于创建以非常普通的方式与对象协作的库。
例如,反射经常在持续存储对象为数据库、XML或其它外部格式的框架中使用。
Javareflection非常有用,它使类和数据结构能按名称动态检索相关信息,并允许在运行着的程序中操作这些信息。
Java的这一特性非常强大,并且是其它一些常用语言,如C、C++、Fortran或者Pascal等都不具备的。
但反射有两个缺点。
第一个是性能问题。
用于字段和方法接入时反射要远慢于直接代码。
性能问题的程度取决于程序中是如何使用反射的。
如果它作为程序运行中相对很少涉及的部分,缓慢的性能将不会是一个问题。
即使测试中最坏情况下的计时图显示的反射操作只耗用几微秒。
仅反射在性能关键的应用的核心逻辑中使用时性能问题才变得至关重要。
许多应用中更严重的一个缺点是使用反射会模糊程序内部实际要发生的事情。
程序人员希望在源代码中看到程序的逻辑,反射等绕过了源代码的技术会带来维护问题。
反射代码比相应的直接代码更复杂,正如性能比较的代码实例中看到的一样。
解决这些问题的最佳方案是保守地使用反射——仅在它可以真正增加灵活性的地方——记录其在目标类中的使用。
一下是对应各个部分的例子:
具体的应用:
1、模仿instanceof运算符号
classA{}
publicclassinstance1{
publicstaticvoidmain(Stringargs[])
{
Classcls=Class.forName("
A"
booleanb1
=cls.isInstance(newInteger(37));
System.out.println(b1);
booleanb2=cls.isInstance(newA());
System.out.println(b2);
catch(Throwablee){
2、在类中寻找指定的方法,同时获取该方法的参数列表,例外和返回值
publicclassmethod1{
privateintf1(
Objectp,intx)throwsNullPointerException
if(p==null)
thrownewNullPointerException();
returnx;
method1"
Methodmethlist[]
=cls.getDeclaredMethods();
methlist.length;
i++)
Methodm=methlist[i];
name
="
+m.getName());
declclass="
m.getDeclaringClass());
Classpvec[]=m.getParameterTypes();
for(intj=0;
j<
pvec.length;
j++)
param#"
+j+"
+pvec[j]);
Classevec[]=m.getExceptionTypes();
evec.length;
exc#"
+j
+"
+evec[j]);
returntype="
m.getReturnType());
-----"
3、获取类的构造函数信息,基本上与获取方法的方式相同
publicclassconstructor1{
publicconstructor1()
protectedconstructor1(inti,doubled)
publicstaticvoidmain(Stringargs