• SpringMVC之自定义注解


    一.什么是SpringMVC之自定义注解

    二.Java注解简介

    Java注解分类

    JDK元注解

    三.自定义注解简介

    自定义注解的分类

    四.自定义注解的基本案例

    案例一(获取类与方法上的注解值)

    案例二(获取类属性上的注解属性值)  

    案例三(获取参数修饰注解对应的属性值)

    五.Aop自定义注解的应用

    Mylog前置通知


    一.什么是SpringMVC之自定义注解

    在SpringMVC中,我们可以使用自定义注解来增强代码的可读性和可维护性。通过自定义注解,我们可以在控制器类、方法或参数上添加自己定义的元数据,来标记一些特殊的行为或属性。

    使用自定义注解,我们可以为SpringMVC添加一些额外的功能或限制。例如,我们可以定义一个@LoginRequired注解,用于标记需要登录才能访问的控制器方法。在SpringMVC的拦截器中,我们可以根据该注解判断用户是否已登录,从而决定是否允许继续执行该方法。

    另外,自定义注解还可以用于参数校验、请求参数绑定、权限控制等方面。通过定义合适的注解,我们可以让SpringMVC自动进行参数校验,从而减少手动编写校验代码的工作量。

    要自定义注解,我们需要使用Java的注解定义语法,结合SpringMVC的相关功能,实现自己所需的功能。首先,我们需要定义一个注解类型,使用@interface关键字声明。然后,在注解中定义需要的元数据,并设置相应的默认值。

    接下来,我们可以在控制器类、方法或参数上使用自定义注解,来标记需要增强的行为或属性。我们还可以通过AOP等方式,根据自定义注解的类型,在程序运行时动态地对标记的代码进行处理。

    总而言之,通过自定义注解,我们可以在SpringMVC中扩展自己的功能和特性,使得代码更加灵活可扩展。

    二.Java注解简介

    Java注解分类

    JDK基本注解
    JDK元注解
    自定义注解

    JDK元注解

    @Retention:定义注解的保留策略
    @Retention(RetentionPolicy.SOURCE)             //注解仅存在于源码中,在class字节码文件中不包含
    @Retention(RetentionPolicy.CLASS)              //默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得,
    @Retention(RetentionPolicy.RUNTIME)            //注解会在class字节码文件中存在,在运行时可以通过反射获取到

    @Target:指定被修饰的Annotation可以放置的位置(被修饰的目标)
    @Target(ElementType.TYPE)                      //接口、类
    @Target(ElementType.FIELD)                     //属性
    @Target(ElementType.METHOD)                    //方法
    @Target(ElementType.PARAMETER)                 //方法参数

    @Target(ElementType.CONSTRUCTOR)               //构造函数
    @Target(ElementType.LOCAL_VARIABLE)            //局部变量
    @Target(ElementType.ANNOTATION_TYPE)           //注解
    @Target(ElementType.PACKAGE)                   //包
    注:可以指定多个位置,例如:
    @Target({ElementType.METHOD, ElementType.TYPE}),也就是此注解可以在方法和类上面使用

    @Inherited:指定被修饰的Annotation将具有继承性

    @Documented:指定被修饰的该Annotation可以被javadoc工具提取成文档.

    三.自定义注解简介

    自定义注解的分类

    自定义注解可分为三类:元注解、标记注解和单值注解。

    1. 元注解(Meta-Annotation):元注解用于注解其他注解,可以对注解本身进行描述和限制。Java提供了四种元注解,分别是:

      • @Retention:指定注解的生命周期,可以是源代码级别(SOURCE)、编译时(CLASS)或运行时(RUNTIME)。
      • @Target:指定注解的使用范围,可以是类、方法、字段等。
      • @Documented:指定注解是否包含在JavaDoc中。
      • @Inherited:指定注解是否可被继承。
    2. 标记注解(Marker Annotation):标记注解也称为无成员注解,它没有定义任何成员变量。标记注解的存在本身就代表一种信息,可以被用于标识目标对象的特殊性或属性。例如,JUnit中的@Test注解就是一个标记注解。

    3. 单值注解(Single Value Annotation):单值注解是最简单的一种注解,它只包含一个成员变量。我们可以为注解定义一个成员变量,并为其赋予默认值。在使用注解时,可以根据需要给成员变量赋值。例如,Spring中的@Autowired注解就是一个单值注解,它用于自动注入依赖对象。

    需要注意的是,以上分类是一种常见的方式,实际上可以根据需求和设计的需要进行灵活扩展和组合。通过合理使用自定义注解,可以提高代码的可读性、可维护性和扩展性。

    四.自定义注解的基本案例

    案例一(获取类与方法上的注解值)

    编写定义常量TranscationModel类

    1. package com.xy.annotation.demo1;
    2. public enum TranscationModel {
    3. //定义常量
    4. Read, Write, ReadWrite
    5. }

    编写MyAnnotation1

    1. package com.xy.annotation.demo1;
    2. import java.lang.annotation.*;
    3. /**
    4. * MyAnnotation1注解可以用在类、接口、属性、方法上
    5. * 注解运行期也保留
    6. * 不可继承
    7. */
    8. @Target({ElementType.TYPE, ElementType.FIELD,ElementType.METHOD})
    9. @Retention(RetentionPolicy.RUNTIME)
    10. @Documented
    11. public @interface MyAnnotation1 {
    12. String name();
    13. }

    编写MyAnnotation2

    1. package com.xy.annotation.demo1;
    2. import java.lang.annotation.*;
    3. /**
    4. * MyAnnotation2注解可以用在方法上
    5. * 注解运行期也保留
    6. * 不可继承
    7. */
    8. @Target(ElementType.METHOD)
    9. @Retention(RetentionPolicy.RUNTIME)
    10. @Documented
    11. public @interface MyAnnotation2 {
    12. TranscationModel model() default TranscationModel.ReadWrite;
    13. }

    编写MyAnnotation3

    1. package com.xy.annotation.demo1;
    2. import java.lang.annotation.*;
    3. /**
    4. * @author xy集团
    5. * @site www.javaxl.com
    6. *
    7. * MyAnnotation3注解可以用在方法上
    8. * 注解运行期也保留
    9. * 可继承
    10. */
    11. @Target(ElementType.METHOD)
    12. @Retention(RetentionPolicy.RUNTIME)
    13. @Inherited
    14. @Documented
    15. public @interface MyAnnotation3 {
    16. TranscationModel[] models() default TranscationModel.ReadWrite;
    17. }

    编写测试类Demo1

    1. package com.xy.annotation.demo1;
    2. /**
    3. * @author xy集团
    4. * @site www.javaxl.com
    5. *
    6. * 获取类与方法上的注解值
    7. */
    8. @MyAnnotation1(name = "abc")
    9. public class Demo1 {
    10. @MyAnnotation1(name = "xyz")
    11. private Integer age;
    12. @MyAnnotation2(model = TranscationModel.Read)
    13. public void list() {
    14. System.out.println("list");
    15. }
    16. @MyAnnotation3(models = {TranscationModel.Read, TranscationModel.Write})
    17. public void edit() {
    18. System.out.println("edit");
    19. }
    20. }

    编写测试类Demo1Test

    1. package com.xy.annotation.demo1;
    2. import org.junit.Test;
    3. /**
    4. * @author xy集团
    5. * @site www.javaxl.com
    6. */
    7. public class Demo1Test {
    8. @Test
    9. public void list() throws Exception {
    10. // 获取类上的注解
    11. MyAnnotation1 annotation1 = Demo1.class.getAnnotation(MyAnnotation1.class);
    12. System.out.println(annotation1.name());//abc
    13. // 获取方法上的注解
    14. MyAnnotation2 myAnnotation2 = Demo1.class.getMethod("list").getAnnotation(MyAnnotation2.class);
    15. System.out.println(myAnnotation2.model());//Read
    16. // 获取属性上的注解
    17. MyAnnotation1 myAnnotation1 = Demo1.class.getDeclaredField("age").getAnnotation(MyAnnotation1.class);
    18. System.out.println(myAnnotation1.name());// xyz
    19. }
    20. @Test
    21. public void edit() throws Exception {
    22. MyAnnotation3 myAnnotation3 = Demo1.class.getMethod("edit").getAnnotation(MyAnnotation3.class);
    23. for (TranscationModel model : myAnnotation3.models()) {
    24. System.out.println(model);//Read,Write
    25. }
    26. }
    27. }

    测试list结果

     测试edit结果

    案例二(获取类属性上的注解属性值)  

    编写TestAnnotation类

    1. package com.xy.annotation.demo2;
    2. import java.lang.annotation.ElementType;
    3. import java.lang.annotation.Retention;
    4. import java.lang.annotation.RetentionPolicy;
    5. import java.lang.annotation.Target;
    6. /**
    7. * @author xy集团
    8. * @site www.javaxl.com
    9. */
    10. //@Retention(RetentionPolicy.SOURCE)
    11. @Retention(RetentionPolicy.RUNTIME)
    12. @Target(ElementType.FIELD)
    13. public @interface TestAnnotation {
    14. String value() default "默认value值";
    15. String what() default "这里是默认的what属性对应的值";
    16. }

    编写Demo2测试类

    1. package com.xy.annotation.demo2;
    2. /**
    3. * @author xy集团
    4. * @site www.javaxl.com
    5. *
    6. * 获取类属性上的注解属性值
    7. */
    8. public class Demo2 {
    9. @TestAnnotation(value = "这就是value对应的值_msg1", what = "这就是what对应的值_msg1")
    10. private static String msg1;
    11. //当没有在注解中指定属性名,那么就是value
    12. @TestAnnotation("这就是value对应的值1")
    13. private static String msg2;
    14. @TestAnnotation(value = "这就是value对应的值2")
    15. private static String msg3;
    16. @TestAnnotation(what = "这就是what对应的值")
    17. private static String msg4;
    18. }

    编写Demo2Tes测试类

    1. package com.xy.annotation.demo2;
    2. import org.junit.Test;
    3. /**
    4. * @author xy集团
    5. * @site www.javaxl.com
    6. */
    7. public class Demo2Test {
    8. @Test
    9. public void test1() throws Exception {
    10. TestAnnotation msg1 = Demo2.class.getDeclaredField("msg1").getAnnotation(TestAnnotation.class);
    11. System.out.println(msg1.value());
    12. System.out.println(msg1.what());
    13. }
    14. @Test
    15. public void test2() throws Exception{
    16. TestAnnotation msg2 = Demo2.class.getDeclaredField("msg2").getAnnotation(TestAnnotation.class);
    17. System.out.println(msg2.value());
    18. System.out.println(msg2.what());
    19. }
    20. @Test
    21. public void test3() throws Exception{
    22. TestAnnotation msg3 = Demo2.class.getDeclaredField("msg3").getAnnotation(TestAnnotation.class);
    23. System.out.println(msg3.value());
    24. System.out.println(msg3.what());
    25. }
    26. @Test
    27. public void test4() throws Exception{
    28. TestAnnotation msg4 = Demo2.class.getDeclaredField("msg4").getAnnotation(TestAnnotation.class);
    29. System.out.println(msg4.value());
    30. System.out.println(msg4.what());
    31. }
    32. }

    测试test1结果

    测试test2结果

    测试test3结果

    测试test4结果

    案例三(获取参数修饰注解对应的属性值)

    编写IsNotNull类

    1. package com.xy.annotation.Demo3;
    2. import java.lang.annotation.*;
    3. /**
    4. * @author xy集团
    5. * @site www.javaxl.com
    6. *
    7. * 非空注解:使用在方法的参数上,false表示此参数可以为空,true不能为空
    8. */
    9. @Documented
    10. @Target({ElementType.PARAMETER})
    11. @Retention(RetentionPolicy.RUNTIME)
    12. public @interface IsNotNull {
    13. boolean value() default false;
    14. }

     编写Demo3测试类

    1. package com.xy.annotation.Demo3;
    2. /**
    3. * @author xy集团
    4. * @site www.javaxl.com
    5. *
    6. * 获取参数修饰注解对应的属性值
    7. */
    8. public class Demo3 {
    9. public void hello1(@IsNotNull(true) String name) {
    10. System.out.println("hello:" + name);
    11. }
    12. public void hello2(@IsNotNull String name) {
    13. System.out.println("hello:" + name);
    14. }
    15. }

     编写Demo3Test测试类

    1. package com.xy.annotation.Demo3;
    2. import org.junit.Test;
    3. import java.lang.reflect.Method;
    4. import java.lang.reflect.Parameter;
    5. /**
    6. * @author xy集团
    7. * @site www.javaxl.com
    8. */
    9. public class Demo3Test {
    10. @Test
    11. public void hello1() throws Exception {
    12. Demo3 demo3 = new Demo3();
    13. for (Parameter parameter : demo3.getClass().getMethod("hello1", String.class).getParameters()) {
    14. IsNotNull annotation = parameter.getAnnotation(IsNotNull.class);
    15. if(annotation != null){
    16. System.out.println(annotation.value());//true
    17. }
    18. }
    19. }
    20. @Test
    21. public void hello2() throws Exception {
    22. Demo3 demo3 = new Demo3();
    23. for (Parameter parameter : demo3.getClass().getMethod("hello2", String.class).getParameters()) {
    24. IsNotNull annotation = parameter.getAnnotation(IsNotNull.class);
    25. if(annotation != null){
    26. System.out.println(annotation.value());//false
    27. }
    28. }
    29. }
    30. @Test
    31. public void hello3() throws Exception {
    32. // 模拟浏览器传递到后台的参数 解读@requestParam
    33. String name = "zs";
    34. Demo3 demo3 = new Demo3();
    35. Method method = demo3.getClass().getMethod("hello1", String.class);
    36. for (Parameter parameter : method.getParameters()) {
    37. IsNotNull annotation = parameter.getAnnotation(IsNotNull.class);
    38. if(annotation != null){
    39. System.out.println(annotation.value());//true
    40. if (annotation.value() && !"".equals(name)){
    41. method.invoke(demo3,name);
    42. }
    43. }
    44. }
    45. }
    46. }

    测试hello1结果

    测试hello2结果

    测试hello3结果

    五.Aop自定义注解的应用

    Mylog前置通知

    编写Mylog前置通知类

    1. package com.xy.annotation.aop;
    2. import java.lang.annotation.ElementType;
    3. import java.lang.annotation.Retention;
    4. import java.lang.annotation.RetentionPolicy;
    5. import java.lang.annotation.Target;
    6. /**
    7. * @author xy集团
    8. * @site www.javaxl.com
    9. */
    10. @Target(ElementType.METHOD)
    11. @Retention(RetentionPolicy.RUNTIME)
    12. public @interface MyLog {
    13. String desc();
    14. }

    编写MyLogAspect切面类

    1. package com.xy.aspect;
    2. import com.xy.annotation.aop.MyLog;
    3. import org.aspectj.lang.JoinPoint;
    4. import org.aspectj.lang.ProceedingJoinPoint;
    5. import org.aspectj.lang.annotation.Around;
    6. import org.aspectj.lang.annotation.Aspect;
    7. import org.aspectj.lang.annotation.Before;
    8. import org.aspectj.lang.annotation.Pointcut;
    9. import org.aspectj.lang.reflect.MethodSignature;
    10. import org.slf4j.Logger;
    11. import org.slf4j.LoggerFactory;
    12. import org.springframework.stereotype.Component;
    13. import java.util.Arrays;
    14. /**
    15. * @author xy集团
    16. * @site www.javaxl.com
    17. */
    18. @Component
    19. @Aspect
    20. public class MyLogAspect {
    21. private static final Logger logger = LoggerFactory.getLogger(MyLogAspect.class);
    22. /**
    23. * 只要用到了com.javaxl.p2.annotation.springAop.MyLog这个注解的,就是目标类
    24. */
    25. @Pointcut("@annotation(com.xy.annotation.aop.MyLog)")
    26. private void MyValid() {
    27. }
    28. // @Before("MyValid()")
    29. // public void before(JoinPoint joinPoint) {
    30. // MethodSignature signature = (MethodSignature) joinPoint.getSignature();
    31. // logger.debug("[" + signature.getName() + " : start.....]");
    32. // System.out.println("[" + signature.getName() + " : start.....]");
    33. //
    34. // MyLog myLog = signature.getMethod().getAnnotation(MyLog.class);
    35. // logger.debug("【目标对象方法被调用时候产生的日志,记录到日志表中】:"+myLog.desc());
    36. // System.out.println("【目标对象方法被调用时候产生的日志,记录到日志表中】:" + myLog.desc());
    37. // }
    38. @Around("MyValid()")
    39. public Object doAround(ProceedingJoinPoint pjp) throws Throwable {
    40. long startTime = System.currentTimeMillis();
    41. System.out.println(pjp.getTarget());//获取目标方法
    42. System.out.println(pjp.getThis());//
    43. Object[] args = pjp.getArgs();//获取参数
    44. System.out.println(Arrays.toString(args));//输出参数
    45. Object ob = pjp.proceed();//获取方法返回值
    46. System.out.println(ob);//输出返回值
    47. logger.info("耗时 : " + (System.currentTimeMillis() - startTime));
    48. return ob;
    49. }
    50. }

     编写LogController类

    1. package com.xy.web;
    2. import com.xy.annotation.aop.MyLog;
    3. import org.springframework.stereotype.Component;
    4. import org.springframework.stereotype.Controller;
    5. import org.springframework.web.bind.annotation.RequestMapping;
    6. /**
    7. * @author xy集团
    8. * @site www.javaxl.com
    9. */
    10. @Controller
    11. public class LogController {
    12. @RequestMapping("/mylog")
    13. @MyLog(desc = "这是结合spring aop知识,讲解自定义注解应用的一个案例")
    14. public void testLogAspect(){
    15. System.out.println("这里随便来点啥");
    16. }
    17. }

    测试结果(先走前置通知)

      测试结果

  • 相关阅读:
    Qt学习04 Hello Qt
    面向城市巡防的多无人机协同航迹规划
    SpringMVC学习笔记(三)
    进程状态
    打造硬核敲门砖——简历
    Springboot毕设项目绿色生鲜5954z(java+VUE+Mybatis+Maven+Mysql)
    仿热血江湖游戏类45method_3、method_4
    前端面试常见问题总结
    电力系统规划学习笔记(长期更新)
    基于HTML+CSS+JavaScript制作简单的大学生网页设计——关于我的家乡湖南网页设计主题
  • 原文地址:https://blog.csdn.net/2201_75455485/article/details/132887991