java 反射 annotation_Java 反射机制及Annotation

转自:http://justjavac.iteye.com/blog/714654

Java 反射是 Java 语言的一个很重要的特征。 它允许运行中的 Java 程序对自身进行检查,并能直接操作程序的内部属性。例如,使用 它能获得 Java 类中各成员的名称并显示出来。

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

a .在运行时判断任意一个对象所属的类。

b .在运行时构造任意一个类的对象。

c .在运行时判断任意一个类所具有的成员变量和方法。

d .在运行时调用任意一个对象的方法。

在 JDK 中,主要由以下类来实现 Java 反射机制,这些类在 java.lang.reflect 包中:

Class 类:代表一个类。

Field 类:代表类的成员变量(成员变量也 称为类的属性)。

Method 类:代表类的方法。

Constructor 类:代表类的构造方法。

Array 类:提供了动态创建数组,以及访问数组的元素的静态方法。

下面写了一个程序:设计了一个 POJO 类。所谓 POJO 类,本人粗浅的理解即和 JavaBean 类似,只有字段和 setter/getter 方法。然后在主函数通过反射,在控制台打印该 POJO 类的所有字段和方法。

本人设计的 POJO 类为 WorkerPOJO.java ,然后另一个测试类为 POJOReflection.java ,在 main 函数中负责打印该类的所有字段和方法。程序见下:

WorkerPOJO.java :

packagecom.xpec.landon.trainjava.annotation;/*** POJO类,和JavaBean相似

*@authorlvwenyong

**/

public classWorkerPOJO{privateString name;private intage;/*** 用Annotation修饰

*@return姓名*/@WorkerPOJOAnnotation(name= "landon",age = 22)publicString getName() {returnname;

}public voidsetName(String name) {this.name =name;

}public intgetAge() {returnage;

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

}

}

POJOReflection.java :

packagecom.xpec.landon.trainjava.annotation;importjava.lang.reflect.Field;importjava.lang.reflect.Modifier;importjava.lang.reflect.Method;/*** 运用Java的反射机制,输出POJO类的字段和方法(新增了Annotation的修饰)

*@authorlvwenyong

**/

public classPOJOReflectionTest {public static voidmain(String[] args)

{try{//加载WorkPOJO,注意这里一定要写全类名,包括包名,因为包名是类名的一部分

Class pojo = Class.forName("com.xpec.landon.trainjava.annotation.WorkerPOJO");//获取域的数组

Field []fieldList =pojo.getDeclaredFields();//获取方法的数组

Method []methodList =pojo.getDeclaredMethods();

System.out.println("WorkerPOJO类的所有字段:");

System.out.println("修饰符" + " " + "类型" + " " + "字段名");for(int i = 0;i < fieldList.length;i++)

{

Field field=fieldList[i];//用下面的形式获取具体的修饰符

System.out.println(Modifier.toString(field.getModifiers()) + " " + field.getType() + " " +field.getName());

}

System.out.println();

System.out.println("WorkerPOJO类的所有方法(不包括annotation修饰的方法):");for(int j = 0;j < methodList.length;j++)

{

Method method=methodList[j];//判断方法是否被Annotation修饰

boolean methodAnnotation = method.isAnnotationPresent(WorkerPOJOAnnotation.class);//如果被annotation修饰,则过滤掉该方法,即不输出

if(methodAnnotation)

{continue;

}//获取方法参数列表

Class parameters[] =method.getParameterTypes();

System.out.print(Modifier.toString(method.getModifiers())+ " " + method.getReturnType() + " " + method.getName() + " (");for(int k = 0;k < parameters.length;k++)

{

System.out.print(parameters[k].toString());

}

System.out.println(")");

}

}catch(ClassNotFoundException exception1)

{

exception1.printStackTrace();

}

}

}

下面是程序的一个运行截图:

0228b885f2ec2141bbf90c0279ce1c7e.png

可以看到,在WorkerPOJO类中引入了Annotation 。

下面,我们详细介绍一下Annotation:

在使用 JUnit4 中,我们可以看到在每个测试方法前面都有一个 @Test 标记,这就是传说中的 Annotation 。

Annotation 提供了一条与程序元素关联任何信息或者任何元数据( metadata )的途径。从某些方面看,annotation 就像修饰符一样被使用,并应用于包、类型、构造方法、方法、成 员变量、参数、本地变量的声明中。这些信息被存储在 annotation 的 “name=value” 结构对中。 annotation 类型是一种接 口,能够通过 java反射 API 的方式提供对其信息的访问。

annotation 能被用来为某个程序元素(类、方法、成员变量等)关联任何的信 息。需要注意的是,这里存在着一个基本的潜规则: annotaion 不能影响程序代码的执行,无论增加、删除 annotation ,代码都始终如一的执行。另外,尽管一些 annotation 通 过 java 的反射 api 方法在运行时被访问,而 java 语言解释器在工作时忽略了这些 annotation 。正是由于 java 虚拟机忽略了 annotation ,导致了 annotation 类型在代码中是 “ 不起作用 ” 的;只有通过某种配套的工具才会对 annotation 类型中的信息进行访问和处理。

annotation 是与一个程序元素相关联信息或者元数据的标注。它从不影响

ava 程序的执行,但是对例如编译器警告或者像文档生成器等辅助工具产 生影响。

我的理解是: Annotation 是继承自 java.lang.annotation.Annotation的类,用于向程序分析工具或虚拟机提供 package class field methed 等方面的信息,它和其他类没什么区别 , 除了使用方式。

下面写了一个简单的Annotation:WorkerPOJOAnnotation.java

packagecom.xpec.landon.trainjava.annotation;importjava.lang.annotation.Documented;importjava.lang.annotation.ElementType;importjava.lang.annotation.Retention;importjava.lang.annotation.RetentionPolicy;importjava.lang.annotation.Target;/*** 修饰WorkerPOJO类方法的一个Annotation

*@authorlvwenyong

**/@Target(ElementType.METHOD)

@Retention(RetentionPolicy.RUNTIME)

@Documentedpublic @interfaceWorkerPOJOAnnotation {

String name();intage();

}

其 中 @Target 里面的 ElementType 是用来指定 Annotation 类型可以用在哪一些元素上的,包括 TYPE(类型), METHOD (方法), FIELD (字段), PARAMETER (参数)等。其中 TYPE 是指可以用在Class,Interface 等类型上。下面给出用 jad 反编译 出的 ElementType 中的静态变量的截图:

343393fcc5531c963198564b81300a12.png

另外 @Retention 中的 RetentionPolicy 是指 Annotation 中的信息保留方式, 分别是 SOURCE,CLASS和 RUNTIME. SOURCE 代表的是这个 Annotation 类型的信息只会保留在程序源码里,源码如果经过了编译之后, Annotation 的数据就会消失 , 并不会保留在编译好的 .class 文件里面。 ClASS 的意思是这个 Annotation类型的信息保留在程序源码里 , 同时也会保留在编译好的 .class 文件里面 , 在执行的时候,并不会把这一些信息加载到虚拟机 (JVM) 中去 . 注意一下,当你没有设定一个 Annotation 类型的 Retention 值时,系统默认值是CLASS. 第三个 , 是 RUNTIME, 表示在源码、编译好的 .class 文件中保留信息,在执行的时候会把这一些信息加载到 JVM 中去的.

下面给出用 jad 反编译出的 RetentionPolicy 中的静态变量的截图

81c8579eb1f19e8cae2f11c219c7ab38.png

最后的一个 Annotation@Documented 是指目的就是让这一个 Annotation 类型的信息能够显示在 Java API 说明文档上。

下面将上面自己设计的 WorkerPOJOAnnotation 应用在了 WorkerPOJO 类的一个方法前面:

1d46465b341d7fc10d34338bf32ed994.png

然后在控制台输出了没有被 Annotation 注释的字段和方法,运行后可以看到不包括 getName 方法。

最后我们可以用Junit4 书写一个测试用例: POJOReflectionJunit4Test .java

packagecom.xpec.landon.trainjava.annotation;importjava.lang.reflect.Field;importjava.lang.reflect.Method;importjavax.activation.FileDataSource;importjunit.framework.Assert;importorg.junit.After;importorg.junit.Before;importorg.junit.Test;/*** 关于Java反射以及Annotation的一个TestCase

*@authorlvwenyong

**/

public classPOJOReflectionJunit4Test {privateClass pojo;privateField []fieldList;privateMethod[] methodList;

@Beforepublic void setUp() throwsException {//加载类WorkPOJO

pojo = Class.forName("com.xpec.landon.trainjava.annotation.WorkerPOJO");//获取域的数组

fieldList =pojo.getDeclaredFields();//获取方法的数组

methodList =pojo.getDeclaredMethods();

}//测试字段和方法的个数

@Testpublic voidtestSize()

{

Assert.assertEquals(2, fieldList.length);

Assert.assertEquals(4, methodList.length);

}//测试字段是否带有annotations

@Testpublic voidisFieldAnnotation()

{for(int i = 0;i < fieldList.length;i++)

{

Assert.assertEquals(false, fieldList[i].isAnnotationPresent(WorkerPOJOAnnotation.class));

}

}//测试方法是否带有annotations

@Testpublic voidisMethodAnnotation()

{for(int i = 0;i < methodList.length;i++)

{

Assert.assertEquals(false, methodList[i].isAnnotationPresent(WorkerPOJOAnnotation.class));

}

}

@Afterpublic void tearDown() throwsException {

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值