/**
*
* 一、自定义注解
* 1、使用@interface 来标识
* 2、内部成员变量通常使用value来表示
* 3、可以指定成员变量的默认值 使用 default 来定义
* 4、如果自定义的注解没有 成员变量 表示一个标识的作用
*
* 5、如果注解有成员 在使用自定义的注解时需要给成员赋值,如果有了默认值就可以不用了
* 但是不想用默认值可以自己重新的赋值
* @MyAnotation("hi")
*
* 二、元注解(对现有注解进行修饰的注解,jdk自带的)有4个
* 1) @Target(ElementType.METHOD) 用来说明当前自定义的注解能修饰那些元素,类? 属性? 还是方法等
* TYPE 类、接口、枚举类
* LOCAL_VARIABLE 局部变量
*
* 2) @Retention(RetentionPolicy.SOURCE) 指定 注解的生命周期
* SOURCE 是编译阶段存在 编译好了就没有 注解了
* CLASS 在.class 文件里面存在,但是不会加载到内存里面,默认值
* RUNTIME 会加载到内存里可以通过反射的方式获取到,进而判断注解是干啥的
*
* 3) @Documented 表示所修饰的注解在被javadoc 解析时会保留
*
* 4)@Inherited 他修饰的注解类将具有继承性,如果某个类使用我们自定义的注解那么这个类的子类也具有该注解
*
* 三、通过反射来获取注解信息
*
* Class clazz = Student.class;
* Annotation[] annotations = clazz.getAnnotations();
*
* for (int i = 0; i < annotations.length; i++) {
* System.out.println(annotations[i]);
* }
*
* 四、jdk 8 新特性中的注解
*
* 1)可重复注解
*
*
* 五、自定义注解的使用场景
*
* 实现自定义注解+拦截器,自定义注解+AOP
*
* 六、类型注解
*
* 注解修饰范型、异常、基本变量
*
*
*/
@Inherited
@Retention(RetentionPolicy.RUNTIME)
@Target({TYPE,METHOD,FIELD,PARAMETER,CONSTRUCTOR,LOCAL_VARIABLE})
public @interface MyAnotation {
String value() default "hello"; //注意这里是成员属性,并提供默认的值
}
public class SourceAccessInterceptor implements HandlerInterceptor {
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("进入拦截器了");
// 反射获取方法上的LoginRequred注解
HandlerMethod handlerMethod = (HandlerMethod)handler;
MyAnotation myAnotation = handlerMethod.getMethod().getAnnotation(MyAnotation.class);
if(myAnotation == null){
return true;
}
// 有LoginRequired注解说明需要登录,提示用户登录
response.setContentType("application/json; charset=utf-8");
response.getWriter().print("你访问的资源需要登录");
return false;
}
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
}
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
}
}
@Configuration
public class InterceptorTrainConfigurer implements WebMvcConfigurer {
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new SourceAccessInterceptor()).addPathPatterns("/**");
}
}
@MyAnotation
@RequestMapping("/hello1")
@ResponseBody
public String hello1(){
String name = myService.sayHello("tomcat");
System.out.println(name);
return name;
}

意思是 同样的自定义注解在一个元素上可以用两次
/**
*
* jdk8 之前的写法 @MyAnotations({@MyAnotation("h"),@MyAnotation("h2")})
*
* jdk8 之后的可以这样写 在 1、MyAnotation 上加一个元注解 @Repeatable 成员值是 MyAnotations.class
* 2、MyAnotation 的 @Inherited @Retention 以及 @Target 要和 MyAnotations 一致
*
*/
//@MyAnotations({@MyAnotation("h"),@MyAnotation("h2")})
@MyAnotation("h2")
@MyAnotation("h3")
@Component
public class Yellow {
private int num = 1;
public void get(){
System.out.println("=======yellow=======");
}
public int getNum() {
return num;
}
public void setNum(int num) {
this.num = num;
}
@Override
public String toString() {
return "Yellow{" +
"num=" + num +
‘}‘;
}
}
/**
*
* 类型注解 两个 TYPE_USE,TYPE_PARAMETER JDK8 以后才有的
* 注解修饰范型、修饰别的地方
*
*
*
*
* @param <T>
*/
public class GenericF<@MyAnotation T> {
public void shows() throws @MyAnotation RuntimeException{
ArrayList<@MyAnotation String> list = new ArrayList<>();
int num = (@MyAnotation int)10L;
}
}
原文:https://www.cnblogs.com/gaohq/p/14907590.html