通过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的嵌套数组占位