目录
定义
使用方法
1.Field
2.Method
3.简单实例
定义
JAVA反射机制是在运行状态中,对于任意一个实体类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。
JAVA反射的方法主要有三种:Class,Method,Field。这里主要简单使用一下Method和Field方法。
使用方法
1.Field
获取对象的属性,给对象属性赋新值
(1)获取Field的常用方法:
getFields():获取public类型的属性(也可以获取到父类的public修饰的属性)并返回一个Field数组;
getDeclaredFields(): 获取所有属性(包括public、protected、default、private权限类型,但不能获取继承父类的属性)并返回一个Field数组;
getField(String name):获取特定的public对象属性,name为属性名;
getDeclaredField(String name):
(2)Field对象的常用方法:
getType():返回Field对象的类型
getName():返回Field对象的名字
set(Object obj,Object value):设置对象属性的值,obj为实例对象,value为要为属性设置的新值
2.Method
获取对象方法,执行对象方法
继承的方法(包括重载、重写和隐藏的)会被编译器强制执行,这些方法都无法反射。
(1)获取Method对象的方法
getMethod(String name,Object type):获取特定的public对象方法,name为方法名,type为参数类型;
getDeclaredMethod(String name,Object type):
(2)Method对象的常用方法:
getReturntype():获取Method对象返回类型
getName():返回Method对象的名字
invoke(Object obj,Object value):执行对象方法,obj为实例对象,value为方法参数,没有参数写null
(3)提示
invoke方法的参数和返回类型必须是Object类型的,这就意味着必须进行多次的类型 转化,这样势必会造成代码出错几率。不仅如此,使用反射获得方法指针的代码要比仅仅直接调用方法慢一些。
3.简单实例
公用对象People
public class People {
public String name = "小白";
//类构造方法
public People(String name){
this.name = name;
}
//属性的get,set方法
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class ReflectTest {
public static void main(String[] args) {
//创建一个People类对象
People people = new People("小白");
System.out.println("初始化后name值:"+people.getName());
//初始化
Field attribute = null;
Method method = null;
System.out.println("Field方法执行结果;");
//用Field对象获取people对象的属性信息
try {
//先用getClass方法获取对象实例,再用getField方法获取name属性的Field类对象
attribute = people.getClass().getField("name");
//getName方法获取name属性名
String attrName = attribute.getName();
//getType方法获取name属性类型
String attrType = attribute.getType().toString();
System.out.println("属性名:"+attrName+"\n属性类型:"+attrType+"\n属性值:"+people.getName());
//set方法重新给name属性赋值
attribute.set(people,"大白");
System.out.println("用set方法重新赋值后name值:"+people.getName());
}catch(Exception e){
e.printStackTrace();
}
System.out.println("Method方法执行结果;");
//用Method对象获取people对象的方法信息
try{
//用getMethod方法获取setName方法的Method对象
method = people.getClass().getDeclaredMethod("setName",String.class);
//用Method的getName方法获取setName方法的方法名
String methodName = method.getName();
//用Method的getReturnType方法获取setName方法的返回值类型
String methodType = method.getReturnType().toString();
System.out.println("方法名:"+methodName+"\n方法类型:"+methodType);
//用Method的invoke方法执行该方法
method.invoke(people,"中白");
System.out.println("invoke方法执行后name值:"+people.getName());
}catch(Exception e){
e.printStackTrace();
}
}
}
控制台输出结果
初始化后name值:小白
Field方法执行结果;
属性名:name
属性类型:class java.lang.String
属性值:小白
用set方法重新赋值后name值:大白
Method方法执行结果;
方法名:setName
方法类型:void
invoke方法执行后name值:中白
4.使用泛型编写泛型数组代码
1)首先获取数组的类对象
2)确认它是一个数组
3)使用Class类(只能定义表示数组的类对象)的getComponent方法确定数组对应类型
4)使用Array类中的静态方法newInstance方法构造新的数组,它有两个参数,一个是数组元素类型,一个是数组长度。
5)用System类中的arraycopy方法将原来数组的数据copy进新数组中,变相实现数据扩容。
import java.lang.reflect.Array;
import java.util.Arrays;
public class ReflectTest {
//泛型数组copy正确方法
public static Object goodCopyOf(Object a,int newLength){
Class c1 = a.getClass();
if(!c1.isArray())
return null;
Class componentType = c1.getComponentType();
int length = Array.getLength(a);
Object newArray = Array.newInstance(componentType,newLength);
System.arraycopy(a,0,newArray,0,Math.min(length,newLength));
return newArray;
}
//泛型数组copy错误方法
public static Object[] badCopy(Object[] a, int newLength){
Object[] newArray = new Object[newLength];
System.arraycopy(a,0,newArray,0,Math.min(a.length,newLength));
return newArray;
}
public static void main(String[] args) {
int [] a = {1,2,3};
a = (int[])goodCopyOf(a,10);
System.out.println(Arrays.toString (a));
String b[] = {"a","b","c"};
b = (String[])goodCopyOf(b,10);
System.out.println(Arrays.toString(b));
System.out.println("This is a bad copy method!");
b = (String[])badCopy(b,10);//抛出错误:java.lang.ClassCastException
}
}