高级篇 KZ002.反射读取注解[未封装]
创建自定义注解
package com.hanpang.java;
/** * 注解说明: 方法的文档注释 * * @Author: 胖先生 * @Create: 2016-04-27 10:29 * @Home: http://www.cnblogs.com/pangxiansheng/ */ import java.lang.annotation.Documented; import java.lang.annotation.ElementType; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;
@Documented @Target(ElementType.METHOD) @Inherited @Retention(RetentionPolicy.RUNTIME) public @interface MethodInfo{ String author() default "hanpang"; String date(); int revision() default 1; String comments(); }
|
|
复习一下:
package com.hanpang.java;
import java.io.FileNotFoundException; import java.util.ArrayList; import java.util.List;
/** * 类说明: 完整点的示例 * @Author: 胖先生 * @Create: 2016-04-27 11:24 */
public class AnnotationExample { public static void main(String[] args) { }
@Override //什么意思? @MethodInfo(author = "wukong", comments = "重写了toString方法", date = "2016-04-27", revision = 1) public String toString() { return "Overriden toString method"; }
@Deprecated //什么意思? @MethodInfo(comments = "该方法已经过时了,被放弃了", date = "2016-04-27") public static void oldMethod() { System.out.println("old method, don't use it."); }
@SuppressWarnings({ "unchecked", "deprecation" }) //什么意思? @MethodInfo(author = "bajie", comments = "这里方法里面有警告哟!", date = "2016-04-27", revision = 10) public static void genericsTest() throws FileNotFoundException { List l = new ArrayList(); l.add("abc"); oldMethod(); } }
|
复习搞一段落,那么我们现在来学习关于注解的解析操作,如果有时间,我会写一个关于Excel的操作的工具
Java注解解析
我们将使用Java反射机制从一个类中解析注解,请记住,注解保持性策略应该是RUNTIME,否则它的信息在运行期无效,我们也不能从中获取任何数据。
类注解的解析
package com.hanpang.java;
import java.lang.annotation.*;
/** * 类说明: 定义类的注解 */ @Documented @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.TYPE) public @interface ClassInfo { String value() default ""; String className(); } |
@ClassInfo(value = "标注在类上",className = "-->com.hanpang.java.AnnotationExample") public class AnnotationExample { public static void main(String[] args) { } }
|
类注解 你可以在运行期访问类,方法或者变量的注解信息,下是一个访问类注解的例子: public class AnnotationParsing { public static void main(String[] args) { //1.获取Class Class clz = AnnotationExample.class; //2.获取类的所有注解 Annotation[] annotations = clz.getAnnotations(); //3.进行迭代 for (Annotation annotation : annotations) { //4.判断获取的是否是我们自己写的注解 if (annotation instanceof ClassInfo) { ClassInfo ci = (ClassInfo) annotation; //5.获取属性值,并且进行测试 System.out.println("-->>hanpang<<--value值=" + ci.value() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--className值=" + ci.className() + "," + "当前类=AnnotationParsing.main()");
}
}
} } |
运行结果为: -->>hanpang<<--className值=-->com.hanpang.java.AnnotationExample,当前类=AnnotationParsing.main() |
你还可以像下面这样指定访问一个类的注解:[如果你已经知道你要对哪个注解进行处理] public class AnnotationParsing { public static void main(String[] args) { //1.获取Class Class clz = AnnotationExample.class; //2.获取你想要的注解 Annotation annotation = clz.getAnnotation(ClassInfo.class); //3.判断是否匹配 if(annotation instanceof ClassInfo){ ClassInfo ci = (ClassInfo) annotation; //获取属性值,并且进行测试 System.out.println("-->>hanpang<<--value值=" + ci.value() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--className值=" + ci.className() + "," + "当前类=AnnotationParsing.main()"); }
} } |
方法的注解解析,代码改造如下
package com.hanpang.java; import java.io.FileNotFoundException; import java.util.ArrayList; import java.util.List;
@ClassInfo(value = "标注在类上",className = "-->com.hanpang.java.AnnotationExample") public class AnnotationExample { public static void main(String[] args) { } @Override @MethodInfo(author = "wukong", comments = "Main method", date = "2016-04-27", revision = 1) public String toString() { return "Overriden toString method"; }
@Deprecated @MethodInfo(comments = "deprecated method", date = "2016-04-27") public static void oldMethod() { System.out.println("old method, don't use it."); }
@SuppressWarnings({ "unchecked", "deprecation" }) @MethodInfo(author = "bajie", comments = "Main method", date = "2016-04-27", revision = 10) public static void genericsTest() throws FileNotFoundException { List l = new ArrayList(); l.add("abc"); oldMethod(); }
}
|
方法注解,解析操作: package com.hanpang.java;
import java.lang.annotation.Annotation; import java.lang.reflect.Method;
public class AnnotationParsing { public static void main(String[] args) throws NoSuchMethodException { //1.实例化对象 AnnotationExample ae = new AnnotationExample(); //2.获取方法的对象--->>>这里写的有点简单类,如果是全部方法请使用数组 Method method = ae.getClass().getMethod("toString");//方法名 Annotation[] annotations = method.getAnnotations(); for (Annotation annotation : annotations) { if(annotation instanceof MethodInfo){ MethodInfo mi = (MethodInfo)annotation; System.out.println("-->>hanpang<<--author值=" + mi.author() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--commonts值=" + mi.comments() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--date值=" + mi.date() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--revision=" + mi.revision() + "," + "当前类=AnnotationParsing.main()"); } } } }
|
运行结果为: -->>hanpang<<--commonts值=Main method,当前类=AnnotationParsing.main() -->>hanpang<<--date值=2016-04-27,当前类=AnnotationParsing.main() -->>hanpang<<--revision=1,当前类=AnnotationParsing.main() |
另一种写法: public class AnnotationParsing { public static void main(String[] args) throws NoSuchMethodException { //1.实例化对象 AnnotationExample ae = new AnnotationExample(); //2.获取方法的对象--->>>这里写的有点简单类,如果是全部方法请使用数组 Method method = ae.getClass().getMethod("toString"); Annotation annotation = method.getAnnotation(MethodInfo.class); if(annotation instanceof MethodInfo){ MethodInfo mi = (MethodInfo)annotation; System.out.println("-->>hanpang<<--author值=" + mi.author() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--commonts值=" + mi.comments() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--date值=" + mi.date() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--revision=" + mi.revision() + "," + "当前类=AnnotationParsing.main()"); } } }
|
参数注解解析:
package com.hanpang.java; /** * 新增参数注解 */
import java.lang.annotation.*;
@Documented @Target(ElementType.PARAMETER) @Retention(RetentionPolicy.RUNTIME) public @interface ParameterInfo { String value() default "hanpang"; String desc(); }
|
//加大加粗为新增代码部分
import java.io.FileNotFoundException; import java.util.ArrayList; import java.util.List;
@ClassInfo(value = "标注在类上",className = "-->com.hanpang.java.AnnotationExample") public class AnnotationExample { public static void main(String[] args) { } @Override @MethodInfo(author = "wukong", comments = "Main method", date = "2016-04-27", revision = 1) public String toString() { return "Overriden toString method"; }
@Deprecated @MethodInfo(comments = "deprecated method", date = "2016-04-27") public static void oldMethod(@ParameterInfo(value = "刘文铭",desc = "测试参数注解解析") String user_name) { System.out.println("old method, don't use it."+user_name); }
@SuppressWarnings({ "unchecked", "deprecation" }) @MethodInfo(author = "bajie", comments = "Main method", date = "2016-04-27", revision = 10) public static void genericsTest() throws FileNotFoundException { List l = new ArrayList(); l.add("abc"); oldMethod("悟空"); }
}
|
package com.hanpang.java;
import java.lang.annotation.Annotation; import java.lang.reflect.Method;
public class AnnotationParsing { public static void main(String[] args) throws NoSuchMethodException { //1.实例化对象 AnnotationExample ae = new AnnotationExample(); //2.获取方法对象 Method method = ae.getClass().getMethod("oldMethod",java.lang.String.class); //3.获取参数的注解, 注意这里是二维数组 Annotation[][] parameterAnnotations = method.getParameterAnnotations(); //4. Class[] parameterTypes = method.getParameterTypes(); for (int i = 0; i < parameterAnnotations.length; i++) { Annotation[] annotations = parameterAnnotations[i]; Class parameterType = parameterTypes[i];
//继续迭代 for (Annotation annotation : annotations) { if (annotation instanceof ParameterInfo){ ParameterInfo pi = (ParameterInfo)annotation; System.out.println("-->>hanpang<<--数据类型值=" + parameterType.getName() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--pi.value()值=" + pi.value() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--pi.desc()值=" + pi.desc() + "," + "当前类=AnnotationParsing.main()"); } } } } }
|
运行结果: -->>hanpang<<--数据类型值=java.lang.String,当前类=AnnotationParsing.main() -->>hanpang<<--pi.value()值=刘文铭,当前类=AnnotationParsing.main() -->>hanpang<<--pi.desc()值=测试参数注解解析,当前类=AnnotationParsing.main() |
注意:需要注意的是 Method.getParameterAnnotations()方法返回一个注解类型的二维数组,每一个方法的参数包含一个注解数组 |
变量注解解析:
package com.hanpang.java;
import java.lang.annotation.*;
@Documented @Target(ElementType.FIELD) @Retention(RetentionPolicy.RUNTIME) public @interface FieldInfo { String value() default "hanpang"; String desc(); }
|
@ClassInfo(value = "标注在类上",className = "-->com.hanpang.java.AnnotationExample") public class AnnotationExample { //新增代码 @FieldInfo(value = "方位为属性和字段",desc = "不测试了,累挺!!") public String account = null;
public static void main(String[] args) { } @Override @MethodInfo(author = "wukong", comments = "Main method", date = "2016-04-27", revision = 1) public String toString() { return "Overriden toString method"; }
@Deprecated @MethodInfo(comments = "deprecated method", date = "2016-04-27") public static void oldMethod(@ParameterInfo(value = "刘文铭",desc = "测试参数注解解析") String user_name) { System.out.println("old method, don't use it."+user_name); }
@SuppressWarnings({ "unchecked", "deprecation" }) @MethodInfo(author = "bajie", comments = "Main method", date = "2016-04-27", revision = 10) public static void genericsTest() throws FileNotFoundException { List l = new ArrayList(); l.add("abc"); oldMethod("悟空"); }
}
|
package com.hanpang.java;
import java.lang.annotation.Annotation; import java.lang.reflect.Field; import java.lang.reflect.Method; public class AnnotationParsing { public static void main(String[] args) throws NoSuchMethodException, NoSuchFieldException { //1.获取类 Class clz = AnnotationExample.class; //2.获取属性对象 Field field = clz.getField("account");//clz.getDeclaredField() 没有测试 //3.获取注解数组 Annotation[] annotations = field.getAnnotations(); for (Annotation annotation : annotations) { if(annotation instanceof FieldInfo){ FieldInfo fi = (FieldInfo)annotation; System.out.println("-->>hanpang<<--fi.value()值=" + fi.value() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--fi.desc()值=" + fi.desc() + "," + "当前类=AnnotationParsing.main()"); } }
} }
|
运行结果: -->>hanpang<<--fi.value()值=方位为属性和字段,当前类=AnnotationParsing.main() -->>hanpang<<--fi.desc()值=不测试了,累挺!!,当前类=AnnotationParsing.main() |
package com.hanpang.java;
import java.lang.annotation.Annotation; import java.lang.reflect.Field; import java.lang.reflect.Method;
public class AnnotationParsing { public static void main(String[] args) throws NoSuchMethodException, NoSuchFieldException { //1.获取类 Class clz = AnnotationExample.class; //2.获取属性 注意一定是public的,如果set方法需要单独进行处理 Field field = clz.getField("account");//clz.getDeclaredField() 没有测试 //3.获取注解数组 Annotation annotation = field.getAnnotation(FieldInfo.class); if(annotation instanceof FieldInfo){ FieldInfo fi = (FieldInfo)annotation; System.out.println("-->>hanpang<<--fi.value()值=" + fi.value() + "," + "当前类=AnnotationParsing.main()"); System.out.println("-->>hanpang<<--fi.desc()值=" + fi.desc() + "," + "当前类=AnnotationParsing.main()"); } } }
|