转自: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 :

  1. package com.xpec.landon.trainjava.annotation;
  2. /**
  3. * POJO类,和JavaBean相似
  4. * @author lvwenyong
  5. *
  6. */
  7. public class WorkerPOJO{
  8.   private String name;
  9.   private int age;
  10.  
  11. /**
  12. * 用Annotation修饰
  13. * @return 姓名
  14. */
  15. @WorkerPOJOAnnotation(name = "landon",age = 22)
  16. public String getName() {
  17.   return name;
  18. }
  19. public void setName(String name) {
  20.   this.name = name;
  21. }
  22. public int getAge() {
  23.   return age;
  24. }
  25. public void setAge(int age) {
  26.   this.age = age;
  27. }
  28.  
  29. }

POJOReflection.java :

  1. package com.xpec.landon.trainjava.annotation;
  2. import java.lang.reflect.Field;
  3. import java.lang.reflect.Modifier;
  4. import java.lang.reflect.Method;
  5.  
  6. /**
  7. * 运用Java的反射机制,输出POJO类的字段和方法(新增了Annotation的修饰)
  8. * @author lvwenyong
  9. *
  10. */
  11. public class POJOReflectionTest {
  12. public static void main(String[] args)
  13. {
  14. try
  15. {
  16. //加载WorkPOJO,注意这里一定要写全类名,包括包名,因为包名是类名的一部分
  17. Class pojo = Class.forName("com.xpec.landon.trainjava.annotation.WorkerPOJO");
  18. //获取域的数组
  19. Field []fieldList = pojo.getDeclaredFields();
  20. //获取方法的数组
  21. Method []methodList = pojo.getDeclaredMethods();
  22.  
  23. System.out.println("WorkerPOJO类的所有字段:");
  24. System.out.println("修饰符" + " " + "类型" + " " + "字段名");
  25.  
  26. for(int i = 0;i < fieldList.length;i++)
  27. {
  28. Field field = fieldList[i];
  29. //用下面的形式获取具体的修饰符
  30. System.out.println(Modifier.toString(field.getModifiers()) + " " + field.getType() + " " + field.getName());
  31. }
  32.  
  33. System.out.println();
  34. System.out.println("WorkerPOJO类的所有方法(不包括annotation修饰的方法):");
  35.  
  36. for(int j = 0;j < methodList.length;j++)
  37. {
  38. Method method = methodList[j];
  39. //判断方法是否被Annotation修饰
  40. boolean methodAnnotation = method.isAnnotationPresent(WorkerPOJOAnnotation.class);
  41.  
  42. //如果被annotation修饰,则过滤掉该方法,即不输出
  43. if(methodAnnotation)
  44. {
  45. continue;
  46. }
  47. //获取方法参数列表
  48. Class parameters[] = method.getParameterTypes();
  49.  
  50. System.out.print(Modifier.toString(method.getModifiers()) + " " + method.getReturnType() + " " + method.getName() + " (");
  51.  
  52. for(int k = 0;k < parameters.length;k++)
  53. {
  54. System.out.print(parameters[k].toString());
  55. }
  56.  
  57. System.out.println(")");
  58. }
  59. }
  60. catch(ClassNotFoundException exception1)
  61. {
  62. exception1.printStackTrace();
  63. }
  64.  
  65. }
  66.  
  67. }

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

可以看到,在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

  1. package com.xpec.landon.trainjava.annotation;
  2. import java.lang.annotation.Documented;
  3. import java.lang.annotation.ElementType;
  4. import java.lang.annotation.Retention;
  5. import java.lang.annotation.RetentionPolicy;
  6. import java.lang.annotation.Target;
  7.  
  8. /**
  9. * 修饰WorkerPOJO类方法的一个Annotation
  10. * @author lvwenyong
  11. *
  12. */
  13.  
  14. @Target(ElementType.METHOD)
  15. @Retention(RetentionPolicy.RUNTIME)
  16. @Documented
  17. public @interface WorkerPOJOAnnotation {
  18. String name();
  19. int age();
  20.  
  21. }

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

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

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


    然后在控制台输出了没有被 Annotation 注释的字段和方法,运行后可以看到不包括 getName 方法。 
    最后我们可以用Junit4 书写一个测试用例: POJOReflectionJunit4Test .java

  1. package com.xpec.landon.trainjava.annotation;
  2. import java.lang.reflect.Field;
  3. import java.lang.reflect.Method;
  4.  
  5. import javax.activation.FileDataSource;
  6.  
  7. import junit.framework.Assert;
  8.  
  9. import org.junit.After;
  10. import org.junit.Before;
  11. import org.junit.Test;
  12.  
  13. /**
  14. * 关于Java反射以及Annotation的一个TestCase
  15. * @author lvwenyong
  16. *
  17. */
  18.  
  19. public class POJOReflectionJunit4Test {
  20. private Class pojo;
  21. private Field []fieldList;
  22. private Method[] methodList;
  23.  
  24. @Before
  25. public void setUp() throws Exception {
  26. //加载类WorkPOJO
  27. pojo = Class.forName("com.xpec.landon.trainjava.annotation.WorkerPOJO");
  28. //获取域的数组
  29. fieldList = pojo.getDeclaredFields();
  30. //获取方法的数组
  31. methodList = pojo.getDeclaredMethods();
  32. }
  33.  
  34. //测试字段和方法的个数
  35. @Test
  36. public void testSize()
  37. {
  38. Assert.assertEquals(2, fieldList.length);
  39. Assert.assertEquals(4, methodList.length);
  40. }
  41.  
  42. //测试字段是否带有annotations
  43. @Test
  44. public void isFieldAnnotation()
  45. {
  46. for(int i = 0;i < fieldList.length;i++)
  47. {
  48. Assert.assertEquals(false, fieldList[i].isAnnotationPresent(WorkerPOJOAnnotation.class));
  49. }
  50. }
  51.  
  52. //测试方法是否带有annotations
  53. @Test
  54. public void isMethodAnnotation()
  55. {
  56. for(int i = 0;i < methodList.length;i++)
  57. {
  58. Assert.assertEquals(false, methodList[i].isAnnotationPresent(WorkerPOJOAnnotation.class));
  59. }
  60. }
  61.  
  62. @After
  63. public void tearDown() throws Exception {
  64. }
  65.  
  66. }

Java 反射机制及Annotation的更多相关文章

  1. java基础知识(十一)java反射机制(上)

    java.lang.Class类详解 java Class类详解 一.class类 Class类是java语言定义的特定类的实现,在java中每个类都有一个相应的Class对象,以便java程序运行时 ...

  2. Java反射机制专题

    ·Java Reflection Reflection(反射)是被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方 ...

  3. Java反射机制详解

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

  4. 入门级 JAVA反射机制

    1.什么是反射? Java中的反射机制是Java语言的一个很重要的特性,是Java “动态性” 的重要体现.Java反射机制让我们在程序运行状态中,对于任意一个类,都能知道这个类的所有属性和方法:对于 ...

  5. Java反射机制概念及应用场景

    Java的反射机制相信大家在平时的业务开发过程中应该很少使用到,但是在一些基础框架的搭建上应用非常广泛,今天简单的总结学习一下. 1. 什么是反射机制? Java反射机制是在运行状态中,对于任意一个类 ...

  6. java反射机制基础

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

  7. Java 反射机制(包括组成、结构、示例说明等内容)

    第1部分 Java 反射机制介绍 Java 反射机制.通俗来讲呢,就是在运行状态中,我们可以根据“类的部分已经的信息”来还原“类的全部的信息”.这里“类的部分已经的信息”,可以是“类名”或“类的对象” ...

  8. Java反射机制的使用(全)

    转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/6566957.html  一:反射是什么 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有 ...

  9. Java 反射机制介绍

    参考文章:http://www.cnblogs.com/skywang12345/p/3345205.html Java 反射机制.通俗来讲呢,就是在运行状态中,我们可以根据“类的部分已经的信息”来还 ...

随机推荐

  1. centos安装软件Error: Cannot find a valid baseurl for repo: base

    今天使用yum安装软件,出现下面的提示: Loaded plugins: fastestmirror, refresh-packagekit, securityLoading mirror speed ...

  2. ldconfig deferred processing now taking place

    在ubuntu下面安装软件,安装结束后,提示:ldconfig deferred processing now taking place 到网上查询了一下,大概意思是说:软件安装完了,是否要重启电脑.

  3. Button控件常用api

    加载按钮的纹理.loadTextures (const std::string &normal, const std::string &selected, const std::str ...

  4. docker commit容器

    docker commit 容器ID 镜象REPOSITORY 镜象TAG 如docker commit 52b41c68ac7b registry.lenovows.com:5000/video-a ...

  5. Linux下读取默认MAC地址

    导读MAC(Media Access Control,介质访问控制)计算机通过它来定义并识别网络设备的位置.在嵌入式linux学习中不可避免也会遇到MAC,本文主要描述了如何通过操作OTP来读取嵌入式 ...

  6. 关于Unity3D中Resources动态加载NGUI图片的方法

    在NGUI中有些图片我需要动态进行变更或者加载,怎么办? 首先在项目中创建一个Resources目录,接着把需要的图片放在这里面,可以有子文件夹么?当然可以,文件结构很重要哦~ NGUI加载图片的方法 ...

  7. k-means

    参考:http://www.cnblogs.com/jerrylead/archive/2011/04/06/2006910.html k-means是无监督的聚类算法,比较简单,但包含的思想不简单, ...

  8. OJ 1188 全排列---康托展开

    题目描述 求n的从小到大第m个全排列(n≤20). 输入 n和m 输出 输出第m个全排列,两个数之间有一空格. 样例输入 3 2 样例输出 1 3 2 #include<cstdio> # ...

  9. VMware Snapshot 工作原理

    VMware中的快照是对VMDK在某个时间点的“拷贝”,这个“拷贝”并不是对VMDK文件的复制,而是保持磁盘文件和系统内存在该时间点的状态,以便在出现故障后虚拟机能够恢复到该时间点.如果对某个虚拟机创 ...

  10. java面试总结-(hibernate ibatis struts2 spring)

    说说Hibernate对象的三种状态 Hibernate对象有三种状态,分别是:临时态(Transient). 持久态(Persistent).游离态(Detached). 临时状态:是指从对象通过n ...