• SpringMVC中的自定义注解


    目录

     简介

    注解(Annotation)在Java编程中的作用

    SpringMVC中的自定义注解

    Java注解是什么?

    为什么在Java开发中注解变得如此重要?

    Java注解分类

    1. 标准注解(JDK基本注解)

    2. 自定义注解

    JDK基本注解

    JDK元注解

    自定义注解

    如何使用自定义注解?

    案例1:获取类与方法上的注解值

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

    示例三:获取参数修饰注解对应的属性值

    Aop自定义注解的应用  

     MyLog自定义注解类

    MyLogAspect切面类

    controller层



     简介

    注解(Annotation)在Java编程中的作用

    在Java编程中,注解(Annotation)是一种元数据(metadata)标记,它们提供了对程序元素(如类、方法、字段等)的额外信息。注解通常不直接影响程序的运行,但它们可以被编译器、开发工具和框架等工具利用,以提供更多的语义信息和指导程序的行为。

    以下是注解在Java编程中的一些重要作用:

    1. 提供元数据信息:注解可以附加元数据信息到程序元素上。例如,你可以使用注解来标记一个类是否是一个实体类,或者一个方法是否需要事务管理。这些元数据可以在运行时被读取和处理。

    2. 编译时检查:注解可以被编译器用来检查代码的正确性。例如,@Override注解用于标识子类中的方法覆盖了父类的方法,编译器可以检查是否满足覆盖的条件,如果不满足,则会报错。

    3. 代码生成:许多框架和工具可以根据注解生成代码。例如,JavaEE框架中的JPA(Java Persistence API)可以根据实体类上的注解自动生成数据库表结构。

    4. 文档生成:注解可以用于生成文档。通过工具如Javadoc,你可以使用注解来为类、方法和字段添加描述性信息,这些信息将生成到文档中,帮助其他开发人员理解代码的含义和用法。

    5. 运行时处理:在运行时,可以使用反射机制来读取和处理注解。这使得你能够根据注解的信息来调整程序的行为。例如,Spring框架中的依赖注入就是基于运行时处理注解的一个典型示例。

    6. 框架扩展:注解使得框架能够更加灵活和可扩展。开发者可以使用注解来扩展框架的功能,而不需要修改框架的源代码。

    注解为Java编程带来了更多的灵活性、可读性和可维护性。它们是现代Java编程中不可或缺的一部分,被广泛用于各种领域,从基础的语法检查到大规模应用程序的开发和维护。

    SpringMVC中的自定义注解

    在SpringMVC中,自定义注解是一种非常有用的工具,它们可以帮助开发者更好地管理和控制Web应用程序的行为。以下是关于SpringMVC中自定义注解的重要性和用途的说明:

    1. 控制器映射

    • 重要性:自定义注解可以用于标记控制器类或方法,以定义请求的映射规则,将HTTP请求与相应的控制器方法关联起来。

    • 用途:通过自定义注解,你可以创建自己的路由规则,以便更清晰地定义URL到控制器方法的映射。这样可以提高代码的可读性和维护性。

    2. 参数验证和绑定
    • 重要性:自定义注解可以用于验证和绑定请求参数,确保它们满足特定的条件和格式。

    • 用途:通过自定义注解,你可以在方法参数上添加验证规则,例如校验参数是否为正整数或非空。这有助于保持控制器方法的干净和规范。

    3. 安全性

    • 重要性:自定义注解可以用于实施安全性控制,限制特定用户或角色的访问权限。

    • 用途:通过自定义注解,你可以标记需要身份验证或授权的控制器方法,以确保只有授权用户可以访问敏感资源。

    4. 日志记录
    • 重要性:自定义注解可以用于记录方法的执行,以便进行调试和审计。

    • 用途:通过自定义注解,你可以将方法的执行情况记录到日志中,包括方法的参数、返回值等信息,以便更容易诊断问题。

    5. 自定义业务逻辑
    • 重要性:自定义注解可以用于标记特定的业务逻辑或行为。

    • 用途:通过自定义注解,你可以实现自己的业务规则,例如标记某个方法需要执行特殊的处理逻辑,或者标记方法作为事务边界。

    总之,SpringMVC中的自定义注解是强大的工具,可以帮助开发者更好地组织、管理和控制Web应用程序的行为。它们提供了一种清晰、可扩展和可维护的方式来定义请求映射、参数验证、安全性、日志记录以及自定义业务逻辑。通过合理使用自定义注解,你可以提高代码的可读性,减少重复性代码,以及更好地满足项目的需求。
    

    Java注解是什么?

    Java注解是Java语言的一种特殊类型的元数据,它们可以在Java代码中通过注解标记程序的元素,如类、方法、变量等,以提供额外的信息。注解以 @ 符号为前缀,可以携带一些参数,用于描述被注解的元素的性质、特征或约束。

    Java注解的声明形式类似于接口,但它们使用关键字 @interface 定义。注解可以包含元素,这些元素可以是基本类型、枚举、类、数组等类型。

    为什么在Java开发中注解变得如此重要?

    Java注解变得重要的原因有几个:

    1. 元数据标记和描述性信息:注解提供了一种在源代码中添加元数据和描述性信息的简单方式,可以用于描述程序元素的属性、特征和约束。

    2. 编译时检查和工具处理:注解可以用于编写编译时检查的规则,或者为开发工具提供信息。通过注解,可以实现更严格的代码检查,发现潜在的错误和问题。

    3. 代码生成和自动化:许多框架和工具可以根据注解生成代码,简化开发流程。例如,ORM(Object-Relational Mapping)框架可以根据实体类上的注解自动生成数据库映射。

    4. 框架和库的集成:注解可以用于框架和库中,使开发者可以通过简单的注解配置来定制框架或库的行为。这种可扩展性和灵活性使得开发更加高效和便捷。

    5. 元编程和反射:注解可以在运行时通过反射获取,使得程序可以根据注解信息动态地执行特定逻辑。这种特性为元编程提供了基础。

    总的来说,Java注解为Java开发提供了一种强大的机制,可以帮助开发者更好地组织、描述、扩展和自动化代码。它们是现代Java开发中的重要工具,广泛用于各种框架、库和应用程序的开发和维护中。

    提供一些常见的Java注解示例,如@Override@Deprecated等。

    Java注解分类

    Java注解可以分为两大类:标准注解(JDK基本注解)和自定义注解。下面将对这两大类注解进行分类介绍。

    1. 标准注解(JDK基本注解)

    这些注解是Java语言自身提供的,用于标记和控制程序的行为。它们通常被广泛用于编写Java类、方法和字段,并由Java编译器、IDE和其他工具识别和处理。

    (1) @Override

    • 用途:用于标记子类中的方法覆盖了父类的方法。它可以帮助编译器检查方法是否正确地覆盖了父类方法。

    (2) @Deprecated

    • 用途:用于标记类、方法或字段已经不推荐使用,通常因为它们已经过时或有更好的替代方案。

    (3) @SuppressWarnings

    • 用途:用于抑制编译器生成的警告信息,通常用于临时禁用某些警告。

    (4) @FunctionalInterface

    • 用途:用于标记接口是一个函数式接口,即只有一个抽象方法,可以用于Lambda表达式。

    (5) @SafeVarargs

    • 用途:用于标记方法是安全的可变参数方法,可以抑制相关警告。

    (6) @SuppressWarnings("serial")

    • 用途:用于抑制编译器生成的关于缺少序列化ID的警告。

    2. 自定义注解

    自定义注解是开发者根据自己的需求创建的注解,用于标记和控制程序的行为。自定义注解通常用于框架、库和应用程序的开发,以提供额外的元数据信息和控制逻辑。

    (1) 用途相关的自定义注解

    • 这类注解根据应用的需要,可以用于标记特定的业务逻辑,如事务控制、日志记录、权限检查等。

    (2) 参数验证的自定义注解

    • 这类注解可以用于标记方法参数,以进行参数的验证和约束,如@NonNull、@NotBlank、@Positive等。

    (3) 自定义路由和映射注解

    • 这类注解可以用于标记控制器类和方法,自定义请求的映射规则,如SpringMVC中的@RequestMapping、@GetMapping等。

    (4) 框架和库特定的自定义注解

    • 许多框架和库会定义自己的自定义注解,用于控制其行为和配置,如Spring框架中的@Transactional、@Component等。

    自定义注解使得开发者能够根据具体需求来扩展和增强Java程序的功能,提供了更高的灵活性和可定制性。它们通常需要开发者定义注解的结构和处理逻辑,以适应特定的应用场景。

    详细讨论标准注解,如@Override@Deprecated@SuppressWarnings等,以及它们的用途。

    JDK基本注解

    • JDK基本注解是Java开发工具包(JDK)中提供的一组预定义注解,用于标记代码的特定元素。这些注解包括:

      @Override:用于标记方法覆盖(重写)父类中的方法。当一个方法使用了该注解时,编译器会检查该方法是否正确地覆盖了父类中的方法。
      @Deprecated:用于标记已过时的方法、类或字段。当一个方法或类被标记为过时时,编译器会发出警告,提醒开发者不再推荐使用该方法或类。
      @SuppressWarnings:用于抑制编译器产生的警告信息。可以在方法、类或整个代码块上使用该注解,以忽略特定类型的警告。

    JDK元注解

    • 什么是元注解?

    元注解是用于注解其他注解的注解。换句话说,元注解是一种特殊类型的注解,它们用于对其他注解进行说明、配置和控制。元注解在Java中扮演着非常重要的角色,因为它们允许开发者自定义注解的行为和语义,从而满足特定需求。

    在Java中,有一些内置的元注解,它们用于注解的声明和使用。以下是一些常见的元注解:

    1. @Retention:用于指定注解的保留策略,即注解在编译时、运行时或者仅在源代码中保留。保留策略包括RetentionPolicy.SOURCERetentionPolicy.CLASSRetentionPolicy.RUNTIME

    2. @Target:用于指定注解可以应用的目标元素类型,如类、方法、字段、参数等。目标类型包括ElementType.TYPEElementType.METHODElementType.FIELD等。

    3. @Documented:用于指定注解是否包含在Java文档中。如果一个注解被标记为@Documented,则它的文档信息将包括在生成的文档中。

    4. @Inherited:用于指定注解是否可以被继承。如果一个注解被标记为@Inherited,则它可以被子类继承。

    这些元注解在自定义注解中起着关键作用,它们允许开发者控制注解的保留策略、目标元素类型、文档生成和继承性。通过选择合适的元注解,可以自定义注解的行为和语义,以满足具体的需求。

    介绍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:获取类与方法上的注解值

    MyAnnotation1

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

     MyAnnotation3

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

     Demo1测试类 : 获取类与方法上的注解值

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

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

    TestAnnotation

    1. package com.liao.annotation;
    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. //@Retention(RetentionPolicy.SOURCE)
    7. @Retention(RetentionPolicy.RUNTIME)
    8. @Target(ElementType.FIELD)
    9. public @interface TestAnnotation {
    10. String value() default "默认value值";
    11. String what() default "这里是默认的what属性对应的值";
    12. }

    测试类

    1. package com.liao.annotation.demo2;
    2. import com.liao.annotation.TestAnnotation;
    3. /**
    4. * 获取类属性上的注解属性值
    5. */
    6. public class Demo2 {
    7. @TestAnnotation(value = "这就是value对应的值_msg1", what = "这就是what对应的值_msg1")
    8. private static String msg1;
    9. @TestAnnotation("这就是value对应的值1")
    10. private static String msg2;
    11. @TestAnnotation(value = "这就是value对应的值2")
    12. private static String msg3;
    13. @TestAnnotation(what = "这就是what对应的值")
    14. private static String msg4;
    15. }

    测试

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

    示例三:获取参数修饰注解对应的属性值

    IsNotNull自定义注解

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

     测试:

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

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

    Aop自定义注解的应用  

     MyLog自定义注解类

    1. package com.liao.annotation;
    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. @Target(ElementType.METHOD)
    7. @Retention(RetentionPolicy.RUNTIME)
    8. public @interface MyLog {
    9. String desc();
    10. }

    MyLogAspect切面类

    1. package com.liao.aspect;
    2. import com.liao.annotation.MyLog;
    3. import org.aspectj.lang.JoinPoint;
    4. import org.aspectj.lang.annotation.Aspect;
    5. import org.aspectj.lang.annotation.Before;
    6. import org.aspectj.lang.annotation.Pointcut;
    7. import org.aspectj.lang.reflect.MethodSignature;
    8. import org.slf4j.Logger;
    9. import org.slf4j.LoggerFactory;
    10. import org.springframework.stereotype.Component;
    11. @Component
    12. @Aspect
    13. public class MyLogAspect {
    14. private static final Logger logger = LoggerFactory.getLogger(MyLogAspect.class);
    15. /**
    16. * 只要用到了com.javaxl.p2.annotation.springAop.MyLog这个注解的,就是目标类
    17. */
    18. @Pointcut("@annotation(com.liao.annotation.MyLog)")
    19. private void MyValid() {
    20. }
    21. @Before("MyValid()")
    22. public void before(JoinPoint joinPoint) {
    23. MethodSignature signature = (MethodSignature) joinPoint.getSignature();
    24. logger.debug("[" + signature.getName() + " : start.....]");
    25. System.out.println("[" + signature.getName() + " : start.....]");
    26. MyLog myLog = signature.getMethod().getAnnotation(MyLog.class);
    27. logger.debug("【目标对象方法被调用时候产生的日志,记录到日志表中】:"+myLog.desc());
    28. System.out.println("【目标对象方法被调用时候产生的日志,记录到日志表中】:" + myLog.desc());
    29. }
    30. }

    controller层

    1. package com.liao.web;
    2. import com.liao.annotation.aop.MyLog;
    3. import org.springframework.stereotype.Component;
    4. import org.springframework.stereotype.Controller;
    5. import org.springframework.web.bind.annotation.RequestMapping;
    6. @Controller
    7. public class LogController {
    8. @RequestMapping("/mylog")
    9. @MyLog(desc = "这是结合spring aop知识,讲解自定义注解应用的一个案例")
    10. public void testLogAspect(){
    11. System.out.println("记录日志。。。");
    12. }
    13. }

  • 相关阅读:
    【数据结构】基础:二叉树
    Linux学习-36-文件系统管理-硬盘结构
    git简明指南
    SpringBoot学习小结之数据库版本管理工具Flyway
    用qt编译qmake
    YOLOv5涨点必备!改进损失函数EIoU,SIoU,AlphaIoU,FocalEIoU,Wise-IoU
    9.Flink实时项目之订单宽表
    智能座舱架构与芯片- (9) 音频篇 上
    竞赛选题 深度学习的水果识别 opencv python
    学习-Java继承和多态之对象类型的转换
  • 原文地址:https://blog.csdn.net/liaozhixiangjava/article/details/132909431