您的位置:首页 > 编程语言 > Java开发

Java技术——Java反射机制分析

2016-08-14 21:28 459 查看
1. Java的反射机制

动态语言是指在程序运行时允许改变程序结构或者变量类型,从这个观点看,JAVA和C++一样,都不是动态语言。但Java它却有着一个非常突出的动态相关机制:反射。

Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

Java反射机制主要提供了以下功能:

(1)在运行时判断任意一个对象所属的类;

(2)在运行时构造任意一个类的对象;

(这一条在下面没有展示实例,但在之前写过的一篇Java中创建对象的5种方式中有所介绍,有兴趣的可以参考查看。)

(3)在运行时判断任意一个类所具有的成员变量和方法;

(4)在运行时调用任意一个对象的方法;

(5)生成动态代理。

2. Java反射API

反射API用来生成在当前Java虚拟机中的类、接口或者对象的信息。

Class类:反射的核心类,可以获取类的属性,方法等内容信息。

Field类:Java.lang.reflect.表示类的属性,可以获取和设置类的中属性值。

Method类:Java.lang.reflect。表示类的方法,它可以用来获取类中方法的信息或者执行方法

Construcor类:Java.lang.reflect。表示类的构造方法。



3.反射常见用法

3.1 判断对象是否属于反射得到的类(isInstance)

class S {
}
public class IsInstance {
public static void main(String args[]) {
try {
Class cls = Class.forName("S");  //创建了一个S类的Class对象
boolean b1 = cls.isInstance(new Integer(37));
System.out.println(b1);   //fasle
boolean b2 = cls.isInstance(new S());
System.out.println(b2);   //true
}
catch (Throwable e) {
System.err.println(e);
}
}
}




3.2 获取某个反射类的所有属性字段

/**
* 获取反射类的所有属性字段
*
* @param ownerClass 反射的类
* @return
* @throws Exception
*/
public Field[] getProperty(Class ownerClass) throws Exception {

//获取该类所有属性字段
//Field[] fields = ownerClass.getFields();//只获取public访问权限的属性字段(包括父类的)
Field[] fields = ownerClass.getDeclaredFields();//只获取该类的所有访问权限的属性(不含父类)

//输出所有属性字段
for(int i=0;i<fields.length;i++){
System.out.println("属性:"+fields[i]);
}

return fields;
}


3.3 获取反射类的某个public属性值

/**
* 获取反射类的某个public属性值
*
* @param ownerClass 反射的类
* @param 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);

//输出该属性信息
System.out.println(fieldName+"的属性值:"+property.toString());

return property;

}


3.4 获取反射类的该中的所有方法

/**
* 获取反射类的该中的所有方法
* @param ownerClass 反射的类
* @return
* @throws Exception
*/
public Method[] getMethods(Class ownerClass) throws Exception {

//获取该类所有方法
//Field[] fields = ownerClass.getMethods();//只获取public访问权限的方法(包括父类的)
Method[] methods = ownerClass.getDeclaredMethods();//只获取该类的所有访问权限的方法(不含父类)

//输出所有方法
for(int i=0;i<methods.length;i++){
System.out.println("方法:" +methods[i]);
}

return methods;
}


3.5 执行反射类的该中的某个方法

/**
* 执行反射类的该中的某个方法
* @param ownerClass 反射的类
* @param methodName 方法名
* @return
* @throws Exception
*/
public Object invokeMethod(Object owner,String methodName,Object[] args) throws Exception {

//得到对象所属类
Class ownerClass = owner.getClass();

//获取该类的某个方法
Method method = ownerClass.getMethod(methodName, null);

//执行某个对象的方法
Object result = method.invoke(owner, args);

//输出结果信息
System.out.println("结果返回值:"+ result);

return result;
}




实例演示方法:

/**
* 测试反射常用方法
*/
public void refTest(){
String className = "com.java.reflecttest.Student";

try {

//通过反射机制,使用类装载器,装载该类
Class<?> stu = Class.forName(className);
Object objStu = stu.newInstance();

//获取反射类的所有属性
System.out.println("\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\n");
System.out.println("调用 getProperty 方法,获取Student类的所有属性");
getProperty(stu);

//获取反射类的某个属性值
System.out.println("\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\n");
System.out.println("调用 getProperty 方法,获取Student类的NAME属性值");
getProperty(objStu,"NAME");

//获取反射类的所有方法
System.out.println("\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\n");
System.out.println("调用 getMethods 方法,获取Student类的所有方法");
getMethods(stu);

//执行反射类的getInfo方法
System.out.println("\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\n");
System.out.println("调用 invokeMethod 方法,执行Student类的getInfo方法");
invokeMethod(objStu, "getInfo", null);

} catch (Exception e) {
e.printStackTrace();
}
}

演示所用的Student类:



package com.java.reflecttest;

/**
* 学生信息类
*
* @author Longxuan
*
*/
public class Student {

/**
* 学号
*/
private String stuId ;

/**
* 学号
*/
public String STUID;

/**
* 姓名
*/
private String name ;

/**
* 姓名
*/
public String NAME;

/**
* 年龄
*/
private int age;

/**
* 年龄
*/
public int AGE;

/**
* 班级
*/
private String classid;

public String getStuId() {
return stuId;
}

public void setStuId(String stuId) {
this.stuId = stuId;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

public String getClassid() {
return classid;
}

public void setClassid(String classid) {
this.classid = classid;
}

/**
* 输出学生信息
*/
public void getInfo(){
System.out.println("学生信息:\n学号:"+stuId+"\t姓名:"+name+"\t年龄:"+age+"\t班级:"+classid);
}

/**
* 构造函数
*/
public Student(){
init();
}

/**
* 私有初始化方法
*/
private void init(){
this.name = "张三";
this.stuId ="1001";
this.age = 14;
this.classid = "A001";
this.NAME = name;
this.AGE = age;
this.STUID = stuId;
}

}




运行结果:



4. 反射的效率



在Stackoverflow上认为反射比较慢的程序员主要有如下看法,如果你面试遇到了,可以这样回答:

(1)验证等防御代码过于繁琐,这一步本来在link阶段,现在却在计算时进行验证。

(2)产生很多临时对象,造成GC与计算时间消耗。

(3)由于缺少上下文,丢失了很多运行时的优化,比如JIT(它可以看作JVM的重要评测标准之一)

当然,我个人的看法是,现代JVM也不是非常慢了,它能够对反射代码进行缓存以及通过方法计数器同样实现JIT优化,所以反射不一定慢。更重要的是,很多情况下,你自己的代码才是限制程序的瓶颈。因此,在开发效率远大于运行效率的的基础上,大胆使用反射,放心开发吧。

文章转自:http://blog.csdn.net/xiaoxian8023/article/details/9206055

http://www.jianshu.com/p/f83556bcae59





内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: