JAVA 反射基本使用
类的反射
获取Class对象的三种方法:
- 类名.class
- 对象.getClass()
- Class.forName("要加载的类名")
- method.getDeclaringClass()
根据API写就行了,大致流程就是:(通过Class 取得构造器对象+方法对象)
- 取得Class对象, 用上述方式之一获取特定类的
Class
类,即该类对应的字节码 - 取得构造器对象,调用
Class
对象的getConstructor(Class<?>... parameterTypes)
获取构造方法对象 - 实例化 ,调用是构造方法类
Constructor
的newInstance(Object... initargs)
方法新建对象 - 取得
Method
对象,调用Class
对象的getMethod(String name, Class<?>... parameterTypes)
获取方法对象,parameterTypes可以是Class[]
Method.invoke ,
调用方法对象类Method
的invoke(Object obj, Object... args)
方法,调用对象上相应方法,obj是执行者,args可以是Object[]
用方法的参数类型唯一标识一个方法,依据:方法的重载
通过Method对象获取信息:
- method.invoke(Object obj,Object... args):Object //调用对象上相应方法,obj是执行者,args是参数Object[]
- method.getParameterTypes() //取得参数列表类型
Class<?>[]
- method.getDeclaringClass() //取得此方法声明类Class对象(方法在继承层次中哪个类声明的就返回那个类)
- method.getDeclaringClass().getName() //取得方法声明类名String
- method.getName() //取得方法名String
method.getReturnType()
//取得返回值类型- method.isAnnotationPresent(Class<A> annotationClass):boolean //当前方法是否有对应注解
- method.getClass().getName() //注意这个取得的结果是“Method”,相当于上边的获取Method.Class类,不要和getDeclaringClass()混淆
通过Class对象获取信息:
- clazz.getName()
- clazz.getSimpleName()
- clazz.newInstance() :Object
- clazz.getConstructor(Class<?>
...
parameterTypes); //取得对应public构造函数 - clazz.getDeclaredConstructor(Class<?>
...
parameterTypes) ; //取得对应任意构造函数 - clazz.getDeclaredFields:Field[]
- clazz.getDeclaredField(String name):Field //取得对应名称的成员变量 Field
- clazz.getInterfaces():Class<?> //实现接口列表
Class<?>[]
- clazz.getSuperclass():Class<?> //父类型Class
- clazz.getMethod(String name,Class<?>
...
parameterTypes):Method //获取方法对象,parameterTypes可以是Class<?>[]
- clazz.getMethods():Method[] //获取public方法列表Method[]
- clazz.getAnnotation(Class<A> annotationClass):A object //取得对应A类型的注解@对象,返回值<A extends Annotation>类型对象
- clazz.isAnnotationPresent(Class<A> annotationClass):boolean //当前类是否有对应注解
Annotation为@interface,自定义@interface A 时可以设置参数,取得<A extends Annotation>对象后可以取得参数
通过Field对象获取信息:
- field.set(Object obj, Object value); //设置obj的对应成员变量值
- field.get(Object obj); //取得obj的对应成员变量值
- field.getType():Class<?> //取得field对应成员变量真实类型
- field.setAccessible(boolean) //开启强制访问private等域
反射用例
String aString="sss";
Object bObject=aString;
System.out.println(bObject.getClass().getName()); //java.lang.String
//输出java.lang.String 即使引用是Object ,也能反射出原始类型
数组的反射
下面这个例子主要说明几点:
- 对于元素同类型的数组,同维数组,class一样
- 不同维,class不同
- 不同维的,父类都是Object一样
- 基本类型一维数组不能直接转换为Object[]
java.util.Arrays
的asList
方法API看看
public class ReflectTest {
public static void main(String[] args) {
int [] a1 = new int[]{1,2,3};
int [] a2 = new int[5];
int [][] a3 = new int[2][3];
System.out.println(a1.getClass() == a2.getClass());//true
System.out.println(a1.getClass());//class [I
System.out.println(a3.getClass());//class [[I
System.out.println(a1.getClass().getSuperclass() == a3.getClass().getSuperclass());//true
System.out.println(a2.getClass().getSuperclass());//class java.lang.Object
//下句编译不通过:Error:(15, 42) java: 不可比较的类型: java.lang.Class<capture#1, 共 ? extends int[]>和java.lang.Class<capture#2, 共 ? extends int[][]>
//System.out.println(a1.getClass() == a3.getClass());
Object []b3 = a3;//通过
//下句编译不通过 Error:(17, 24) java: 不兼容的类型: int[]无法转换为java.lang.Object[]
//Object [] b1 = a1;
String s1 = "abc";
System.out.println(Arrays.asList(a1));//[[I@1540e19d]
System.out.println(Arrays.asList(s1));//[abc]
}
}