java基础之反射(reflect)

上周记录了java基础之注解(Annotation),这周他的好基友“反射”如约而至(明明是姗姗来迟~)。

反射的基本概念

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

反射的基石:class类

Class类的各个实例对象分别对应各个类在内存中的字节码,例如TestModel类的字节码,ArrayList类的字节码,等等。

一个类被类加载器加载到内存中,占用一片存储空间,这个空间里面的内容就是类的字节码,不同的类的字节码是不同的,所以他们在内存中的内容是不同的,这一个个空间可分别用一个个对象来表示,这些对象显然具有相同的类型,这个类型就是Class类型

这里一定要分清一个对象的Class类型和对象的内容。 两个不同的对象是可以有相同的Class类型的。
比如说: 下面两个字符串明显不是同一个的对象,但他们的Class类型是一样的:java.lang.String

String str1 = "abc";//字符串1
String str2 = "abcd";//字符串2

反射常用方法

实例化Class类对象

Class clazz = Class.forName("java.lang.String");
Class clazz = new String().getClass();
Class clazz = String.class;

Class中的Constructor类,实例化一个对象

/**获取String对象的所有构造方法*/
Constructor[] constructors = Class.forName("java.lang.String").getConstructors();

/**获取String默认的构造方法生成String对象*/
String str = String.class.newInstance();

/**获取String对象指定的构造方法(通过方法的参数类型,传递参数的Class对象)*/
Constructor constructor = String.class.getConstructor(StringBuffer.class);
String str = (String) constructor.newInstance(new StringBuffer("abc"));

Class中的Field类

/**获取clazz这个Class的所有字段*/
Field[] fields = clazz.getDeclaredFields();

//获取clazz的字段名为“X”的字段
//getField方法只能获取到public字段,getDeclaredField可以获取到所有的字段
Field field = clazz.getDeclaredField("x");

//对于private修饰的字段需要设置Accessible为true
field.setAccessible(true);

//设置和获取field的属性
 field.set(testModel, (String)name.value());
 field.get(obj);

//获取字段的类型
 field.getGenericType()

//判断是否有TestAnnotationName注解
field.isAnnotationPresent(TestAnnotationName.class)
//获取字段上的注解
 field.getAnnotation(annotationClass)

Class中的Method类

获取Method的方法和获取field的方法基本一致

//获取所有方法
Method[] methods = clazz.getDeclaredMethods();

//获取对应方法
Method methodCharAt = clazz.getMethod("charAt", int.class);

//调用方法
methodCharAt.invoke(object,18);

获取一个对象的父类与实现的接口

 // 取得父类
 Class parentClass = clazz.getSuperclass();
 // 获取所有的接口
 Class intes[] = clazz.getInterfaces();

使用反射解析注解

终于说到注解了

声明注解

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TestAnnotationAge{
    int value() default 18;
}

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TestAnnotationName{
    String value() default "aclululu";
}

设置注解

public class TestModel {

    @TestAnnotationName
    private String name;

    @TestAnnotationAge
    private int age;

    /**省略getter setter方法**/
}

解析注解

public static void main(String[] args) throws IllegalArgumentException, IllegalAccessException {
    TestModel testModel = new TestModel();
    System.out.println(testModel.toString());
    System.out.println("通过反射机制将注解信息设置给Model...");
    Class clazz = TestModel.class;
    Field[] fields = clazz.getDeclaredFields();
    for(Field field :fields){
        if(field.isAnnotationPresent(TestAnnotationName.class)){
            TestAnnotationName name = (TestAnnotationName) field.getAnnotation(TestAnnotationName.class);
            field.setAccessible(true);
            field.set(testModel, (String)name.value());
        }
        else if(field.isAnnotationPresent(TestAnnotationAge.class)){
             TestAnnotationAge age= (TestAnnotationAge) field.getAnnotation(TestAnnotationAge.class);
             field.setAccessible(true);
             field.set(testModel, (int)age.value());
        }
    }
    
    System.out.println(testModel.toString());
}

输出结果

TestModel [age=0, name=null]
通过反射机制将注解信息设置给Model...
TestModel [age=18, name=aclululu]

你可能感兴趣的:(java基础之反射(reflect))