通过java的反射机制,程序员可以更深入的控制程序的运行过程。例如,可在程序运行时对象用户输入的信息进行验证,还可以逆向控制程序的执行过程,讲解了反射,另外java还提供了Annotation注解功能,该功能建立在反射机制的基础上,包括定义Annotation类型的方法和程序运行时访问Anntation信息的方法
 
16.1反射
通过java反射机制,可以在程序中访问以及装载到JVM中的java对象的描述实现方法。检查喝修改描述java对象本身的信息的功能 java反射机制功能十分强大,在java.lang.reflect包中提供了该功能的支持
众所周知,所有jvaa类俊继承了Object类在Object类中定义了一个getClass()方法 该方法返回了一个类信息为Class的对象 代码如下:
-  demo1 d1 = new demo1();
-  Class c1 = d1.getClass();//先new一个对象然后使用getClass方法来获取
-  //第二种方法
-  Class c2= demo1.class;
-  第三种方法直接使用forName方法来获取 其中括号内传入的时包名和类名
-  Class c3= Class.forName("com.cr.demo1");
利用Class类的对象demo1,可以访问用来返回该对象的demo1对象的描述信息,可以访问的主要描述信息如下
| 组成部分 | 访问方法 | 返回值类型 | 说明 | 
| 包路径 | getPackage() | Package对象 | 获取该类的存放路径 | 
| 类名称 | getName() | String对象 | 获取该类名称 | 
| 继承类 | getSuperclass | Class对象 | 获取该类的继承类 | 
| 实现接口 | getInterfaces() | Class型数组 | 获取该类实现的所有接口 | 
| 构造方法 | getConstructors() | Constructor型数组 | 获取所有权限为public的构造方法 | 
| getConstructor(Class<?>...parameterTypers) | Constructor对象 | 获取指定构造方法 | |
| getDeclaredConstructors() | Constructor型数组 | 获取所有构造方法 | |
| getDeclaredConstructors(Class<?>...parameterTypers) | Constructor对象 | 获取自动那个构造方法 | |
| 方法 | getMethods() | Method型数组 | 获取所有权限为public的指定方法 | 
| getMethod(String name,Class<?>...parameterTypes) | Method对象 | 获取权限为public的指定方法 | |
| getDeclardMethod() | Method型数组 | 获取所有方法 | |
| getDeclaredMethod(String name,Class<?>...parameterTypes) | Method对象 | 获取指定方法 | |
| 成员变量 | getFields() | Field数组 | 获取所有权限为public的成员变量 | 
| getFields(String name) | Field对象 | 获取权限为public的指定的成员变量 | |
| getDeclaredFields() | Field数组 | 获取所有成员变量 | |
| getDeclaredField(String name) | Field对象 | 获取指定成员变量 | |
| 内部类 | getClasses() | Class型数组 | 获取所有权限为public的内部类 | 
| getDeclaredClasses() | Class型数组 | 获取所有内部类 | |
| 内部类的声明类 | getDeclaringClass() | Class对象 | 如果改类为内部类,则返回她的成员变量否则返回null | 
如上代码的使用方法如下代码所示:
-  package com.cr;
-  import java.lang.reflect.Constructor;
-  import java.lang.reflect.Field;
-  import java.lang.reflect.Method;
-  import java.lang.reflect.Parameter;
-  public class classdemo {
-  public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, NoSuchFieldException {
-  demo1 d1 = new demo1();
-  Class c1 = d1.getClass();
-  Class c2= demo1.class;
-  Class c3= Class.forName("com.cr.demo1");
-  System.out.println(c2.getName());
-  System.out.println(c2.getPackageName());
-  //获取构造方法
-  Constructor[] cs = c2.getConstructors();
-  Constructor c = c2.getConstructor(String.class);
-  System.out.println(c.getName());
-  Parameter[] p = c.getParameters();
-  //获取成员变量
-  Field[] fs = c2.getFields();
-  Field f = c2.getDeclaredField("i");
-  System.out.println(f.getName());
-  //获取成员方法
-  Method[] ms =c2.getMethods();
-  Method m = c2.getMethod("getStr", null);
-  System.out.println(m.getName());
-  m.getExceptionTypes();//获取所有异常
-  }
-  }
内置注解
内置注解有三种
以下三种代码都不会影响我们代码的正常运行 是写个编译器看的
@Override限定重写父类方法 作用范围使用在成员方法上
@SuppresWarnings(警告)抑制编译器警告//作用范围类成员属性 成员方法
@Deprecated 标识已经过时//作用范围类成员属性 成员方法
自定义注解
在定义Annotation类型时,也需要用到来i当以的interface关键字,但需要在interface关键字前加一个@符号,即定义Annotation类型的关键字@interface,这个关键字的隐含意思是继承java.lang.annotation.Annotation接口.如下:
-  public @interface demo{
-  }
枚举类中ElementType中的枚举常量使用方法如下所示:
-  package com.cr;
-  //注解类
-  import java.lang.annotation.ElementType;
-  import java.lang.annotation.Retention;
-  import java.lang.annotation.RetentionPolicy;
-  import java.lang.annotation.Target;
-  @Retention(value =RetentionPolicy.RUNTIME )//在源码时生效
-  @Target(value = {ElementType.CONSTRUCTOR,ElementType.FIELD ,ElementType.METHOD,ElementType.TYPE })
-  //设置什么方法可以使用 如:ElementType.CONSTRUCTOR是构造方法 FIELD 适用于成员变量和枚举常量 METHOD用于方法 TYPE是用于类和接口枚举 Annotation类型
-  public @interface MyAnnotation {
-  String value()default"田";
-  int i ()default 8;
-  }
-  package com.cr;
-  @MyAnnotation(i = 5, value = "联系类")//给某给类注解 //想要给成员变量或者普通方法注解的话 需要把@MyAnnotation放在他的上一行
-  public class demo {
-  @MyAnnotation
-  String str;
-  @MyAnnotation
-  public String getStr() {
-  return str;
-  }
-  }
访问Annotation信息
如果在定义Annotation类型时讲@Retention设置为RetentionPollcy.RUNTIME,那么在运行程序时,通过反射就可以获取相关的Annotation信息 ,如获取构造方法,字段和方法Annotation信息。
Cpmstructor类Field和Method类均继承了isAccessibleObject类,在AccessibleObject中定义了3个关于Annotation的放啊,其中,方法发isAnnotationPresent(Class<?extends Annotation>AnnotationClass)用来查看是否添加了指定类型的Annotation,如果是则返回true 否则返回false;方法getAnnotation(Class<T>AnnotationClass)用来获取指定类型的Annotation如果存在则返回相应的对象,则返回null方法getAnnotations()用来获取所有的Annotation改方法将返回Annotation数组
在constructor类和Method类中还定义了getParameterAnnotations()用来获得所有参数添加Annotation,将以Annotation可u下的二位数组返回,在数组中的素和顺序与声明的顺序相同。如果没有参数则返回一个长度为0的数组,如果存在未添加的Annotation的参数将用一个长度为0的嵌套数组占位