在项目中,有非常多的场景需要用到自定义注解,当然出去面试也必然会被问到spring 如何实现自定义注解。项目中的请求日志,定时任务日志,部分接口限流请求等需要用到自定义注解
data:image/s3,"s3://crabby-images/c6085/c6085e6f43b126b153e0e07f93b9b69010914fa4" alt=""
- 声明注解
package com.coderpwh.note.note;
import java.lang.annotation.*;
/**
* @author coderpwh
* @create 2020-03-06 15:46
* @desc ${DESCRIPTION}
**/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface CoderLog {
String value() default "this is log Are you ok?";
}
@Documented:是否将注解添加到java文档中去
@Retention:该注解的生命的周期:
- SOURCE(编译阶段丢弃,不会写入字节码) ,
- CLASS(类加载阶段,在类加载的时候丢弃,在字节码文件的处理中有用。注解默认使用这种方式。),
- RUNTIME(始终不会丢弃,运行期也保留该注解,因此可以使用反射机制读取该注解的信息。自定义的注解通常使用这种方式)
@Target:注解用到什么地方,ElementType.METHOD 用在方法上面,更多枚举值可以参考ElementType的源码
- 注解的实现
package com.coderpwh.note.note;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
/**
* @author coderpwh
* @create 2020-03-06 16:51
* @desc ${DESCRIPTION}
**/
@Component
@Aspect
public class CoderLogAspect {
@Pointcut("@annotation(com.coderpwh.note.note.CoderLog)")
private void pointcut() {
}
@Before("pointcut() &&@annotation(logger)")
public void advice(CoderLog logger) {
System.out.println(" ---- 哥写的日志内容为:[ " + logger.value() + "] -----");
}
}
- @Componet:将实例注解到容器中
- @Aspect:表示切面容器读取
- @Pointcut: Pointcut是植入Advice的触发条件。每个Pointcut的定义包括2部分,一是表达式,二是方法签名。方法签名必须是 public及void型。可以将Pointcut中的方法看作是一个被Advice引用的助记符,因为表达式不直观,因此我们可以通过方法签名的方式为 此表达式命名。因此Pointcut中的方法只需要方法签名,而不需要在方法体内编写实际代码。
- @Before:标识一个前置增强方法,相当于BeforeAdvice的功能,相似功能的还有
run一下,访问接口:http://localhost:8080/user/get/userInfo?userName=hello
结果如下:
data:image/s3,"s3://crabby-images/b40ad/b40ad51fb2e11f0e73e450d1c825b65f3e1711d1" alt=""
代码地址:https://github.com/coder-PengWenHao/Spring-Boot-Cloud-demos/tree/master/spring%20boots/customNote