Reflection 是 Java 程序开发语言的重要特征之一,是学习Java必须知识点。
Java反射机制主要提供了以下功能:在运行时构造一个类的对象;判断一个类所具有的成员变量和方法;调用一个对象的方法;生成动态代理。反射最大的应用就是框架
Java.lang.Class和Java.lang.reflect包下的API,用于表示或者处理当前JVM中的类,接口和对象。
Java反射的主要功能:
确定一个对象的类
取出类的modifiers,数据成员,方法,构造器,和超类.
找出某个接口里定义的常量和方法说明.
创建一个类实例,这个实例在运行时刻才有名字(运行时间才生成的对象).
取得和设定对象数据成员的值,如果数据成员名是运行时刻确定的也能做到.
在运行时刻调用动态对象的方法.
创建数组,数组大小和类型在运行时刻才确定,也能更改数组成员的值.
Class类是Java反射的基础,Class类表示正在运行的java应用程序中的类和接口。Class只有私有的构造函数。Class类在加载类时由Java虚拟机以及调用类加载器中的 defineClass方法自动创建的。只在内存中存在一份加载类的Class类。
三种方式得到某个Java类的Class类,以java.lang.String为例:
String s = "aaa";
Class cls1 = s.getClass();
Class cls2 = String.class;
Class cls3 = Class.forName("java.lang.String");
if (cls1 == cls2) {
System.out.println("cls1 == cls2");
}
if (cls2 == cls3) {
System.out.println("cls2 == cls3");
}
为什么他们三个会==呢?因为他们描述的都是同一个类java.lang.String类。
Class类还有很多方法,配合java.lang.reflect包下的一些API可以实现更多的功能。
一个javabean
package org.legend.reflect;
public class UserInfo {
private String userName;
private Integer age;
public String school;
public UserInfo() {
}
public UserInfo(String name, Integer age, String shool) {
this.userName = name;
this.age = age;
this.school = shool;
}
public UserInfo(String name, Integer age) {
this.userName = name;
this.age = age;
}
public String getInfo(String n, Integer i) {
return "success" + n + i;
}
public void getMyInfo(String mName, String mBirth, Integer age) {
System.out.println("我是一个来自" + mBirth + "的名叫:" + mName + "的" + age
+ "岁的选手");
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public String getUserName() {
return userName;
}
public void setUserName(String userName) {
this.userName = userName;
}
}
一个操作类:‘
package org.legend.reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* 反射操作类
*/
public class GetMyInfo {
public static void main(String[] args) throws ClassNotFoundException, SecurityException, NoSuchMethodException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
Class mclass = Class.forName("org.legend.reflect.UserInfo");
System.out.println("------------输出类名-----------------------");
System.out.println(mclass.getName());
System.out.println(mclass.getSimpleName()); //基础类的简称
//获取属性(字段)
System.out.println("------------输出所有属性--------------------");
Field[] fileds=mclass.getDeclaredFields(); //得到所有的字段,包括公共,保护,默认(包)和私有变量,但不包括继承的字段。
//Field[] fileds = mclass.getFields(); //得到所有的公共字段。
for(Field field:fileds)
{
String fieldName = field.getName(); //属性名称
int fieldFangWen = field.getModifiers(); //属性访问权限修饰符
Class fieldType = field.getType(); //属性类型
System.out.println(Modifier.toString(fieldFangWen)+" "+fieldType.getSimpleName()+" "+fieldName);
}
//获取方法
System.out.println("------------输出所有方法--------------------");
Method [] methods=mclass.getDeclaredMethods();
//Method[] medthods = mclass.getMethods();
for(Method method:methods)
{
String methodName = method.getName(); //方法名称
int methodFangWen = method.getModifiers(); //访问修饰符
Class methodRetrunType = method.getReturnType();//返回类型
Class [] methodParameter = method.getParameterTypes();//方法的参数列表
System.out.print(Modifier.toString(methodFangWen)+" "+methodRetrunType.getSimpleName()+" "+methodName+"(");
for(int k=0;k<methodParameter.length;k++)
{
String parameterName=methodParameter[k].getSimpleName();
if(k!=methodParameter.length-1)
{
System.out.print(parameterName+" arg"+k+",");
}
else
System.out.print(parameterName+" arg"+k);
}
System.out.println(");");
}
//获取构造
System.out.println("------------输出所有构造器--------------------");
Constructor[]constructors = mclass.getConstructors();
for(Constructor constructor:constructors)
{
String constructorName = constructor.getName();
Class[] constructorParameter = constructor.getParameterTypes();
System.out.print(mclass.getSimpleName()+" "+constructorName.substring(constructorName.lastIndexOf(".")+1, constructorName.length())+"(");
for(int h=0;h<constructorParameter.length;h++)
{
String parameterName = constructorParameter[h].getSimpleName();
if(h!=constructorParameter.length-1)
System.out.print(parameterName+" arg"+h+",");
else
System.out.print(parameterName+" arg"+h);
}
System.out.println(");");
}
//如何执行指定的方法
System.out.println("------------反射执行方法--------------------");
String name ="getMyInfo";
Class[] parameterTypes = new Class[3];
parameterTypes[0]= String.class;
parameterTypes[1]= String.class;
parameterTypes[2]= Integer.class;
Method me = mclass.getDeclaredMethod(name, parameterTypes);
Object obj = mclass.newInstance();
Object[] arg = new Object[3];
arg[0]="范晶晶";
arg[1]="AAA风景区";
arg[2]=new Integer(18);
me.invoke(obj, arg);
}
}
反射的应用很多,很多框架都有用到
spring 的 ioc/di 也是反射…
javaBean和jsp之间调用也是反射…
struts的 FormBean 和页面之间…也是通过反射调用…
JDBC 的 classForName()也是反射…
hibernate的 find(Class clazz) 也是反射…
反射还有一个不得不说的问题,就是性能问题,大量使用反射系统性能大打折扣。怎么使用使你的系统达到最优就看你系统架构和综合使用问题啦,这里就不多说了。。。