一键搞定内网穿透 联行号查询|开户行查询 在线工具箱 藏经阁
当前位置:首页 / 杂记 / 正文
java注解Annotation
推荐查看一下
https://www.jianshu.com/p/bbbc9f4c7887

元注解

元注解:注解里的注解,这些元注解来用对注解的作用进行详细说明。主要有以下几个: @Target、 @Retention、 @Documented、 @Inherited

   @Target

@Target用来说明注解的使用使用范围。 它的主要取值有
ElementType.CONSTRUCTOR //用在构造函数上
ElementType.METHOD //用在类方法上
ElementType.PARAMETER //用于描述参数
ElementType.FIELD, //用在类属性上(包含enum声明)
ElementType.TYPE //用于描述类、接口(包括注解类型) 或enum声明
ElementType.ANNOTATION_TYPE //用于注解类型,用这个一般是注解的注解,如@Target、 @Retention、 @Documented、 @Inherited
idea中打开ElementType源码里有详细的所有取值及英文说明


   @Retention

@Retention表示需要在什么级别保存该注释信息,用于描述注解的生命周期 定义了该Annotation被保留的时间长短:某些Annotation仅出现在源代码中,而被编译器丢弃;而另一些却被编译在class文件中;编译在class文件中的Annotation可能会被虚拟机忽略。 取值有
RetentionPolicy.SOURCE //注解只保留在源文件,当Java文件编译成class文件的时候,注解被遗弃。如果只是做一些检查性的操作,比如 @Override 和 @SuppressWarnings,使用SOURCE 注解
RetentionPolicy.CLASS //注解被保留到class文件,但jvm加载class文件时候被遗弃,这是默认的生命周期。如果要在编译时进行一些预处理操作,比如生成一些辅助代码(如 ButterKnife),就用 CLASS注解
RetentionPolicy.RUNTIME //注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在。一般如果需要在运行时去动态获取注解信息,那只能用 RUNTIME 注解

   @Documented

如果有这个注解,则使用javadoc等工具生成的帮助文档中将会有注解内容。

   @Inherited

说明子类可以继承父类中的该注解

属性

   特点总结

只能用public或默认(default)这两个访问权修饰
String value(); //这里把方法设为defaul默认类型
可以使用default为属性设置默认值,使用时可以不为该属性指定值(此时使用默认值)。
public @interface Autowired {
 boolean required() default true;
}
//使用示例
@Autowired 等同于 @Autowired(required = true)
如果中中含有名为value的属性,在使用时如果只使用value属性的话,可以不写属性名直接指定值。也就是说如果只有一个参数成员,最好把参数名称设为"value"。
public @interface RequestMapping {
 String[] value() default {};
}
//使用示例
@RequestMapping("/info") 等同于 @RequestMapping( )
对于数组类型,当数组中只有一个元素时,可以省略大括号。
//使用示例
@RequestMapping( ) 等同于 @RequestMapping(value={"/info"}) 等同于 @RequestMapping("/info") 等同于 @RequestMapping({"/info"})

   注解属性的可支持数据类型

1.所有基本数据类型(int,float,boolean,byte,double,char,long,short) 2.String类型 3.Class类型 4.enum类型 5.Annotation类型 6.以上所有类型的数组

自定义注解及使用

自定义注解最主要作用是: 1.数据验证 2. SpringMVC的拦截器或者SpringAOP获取添加了注解的方法,在拦截器或者AOP里对注解进行处理完成日志权限等操作。
//自定义注解
@Target({ElementType.METHOD, ElementType.TYPE, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface AnnTest {
 String value() default "";
}

//使用注解
@AnnTest("10")
class A {
 @AnnTest("20")
 private String name;
 @AnnTest("30")
 public String getName() {
  return name;
 }
}
上面自定义注解后使用了它,但实际没有任何意义,因为缺少了注解处理器来获取注解信息。注解处理一般用反射机制来获取注解中的内容(也可以使用apt),一般是通过SpringMVC的拦截器或者SpringAOP获取添加了注解的方法,在拦截器或者AOP里对注解进行处理。

@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
 AnnTest annTest = A.class.getAnnotation(AnnTest.class);
 System.out.println(annTest.value()); //10

 Field fields1 = A.class.getDeclaredField("name");
 AnnTest annTest1 = fields1.getAnnotation(AnnTest.class);
 System.out.println(annTest1.value());//20

 Method method = A.class.getDeclaredMethod("getName", null);
 AnnTest annTest2 = method.getAnnotation(AnnTest.class);
 System.out.println(annTest2.value()); //30
}
使用自定义注解+SpringAOP实现日志记录功能:
//定义
@interface MyAnnLog{}
//使用
@MyAnnLog(operationType="add操作",operation )
//AOP拦截解析
public void after(JoinPoint joinPoint){
 Method[] methods = joinPoint.getTarget().getClass().getMethods();
 for (Method method : methods) {
  MyAnnLog annotation = method.getAnnotation(MyAnnLog.class);
  String operationName = annotation.operationName();
  String operationType = annotation.operationType();
  .....
 }
}