AOP中获取自定义注解的参数值

目录

一、利用注解实现AOP的基本流程

  1.1、创建一个注解,用来注解切点(pointcut)

  1.2、创建一个service,使用上面定义的注解来指定切点

  1.3、创建Aspect,增加业务逻辑

  1.4、创建Spring配置类

  1.5、测试

二、获取自定义注解的参数

  2.1、创建带属性的自定义注解

  2.2、创建service使用带属性的自定义注解

  2.3、创建Aspect的错误示例

  2.4、创建Aspect的正确做法

  2.5、测试

三、总结

 

 

一、利用注解实现AOP的基本流程 

  如果特别熟悉自定义注解实现AOP,可以直接转到第二部分:跳转

  Spring中,可以通过自定义注解的方式来实现AOP,比较简单,流程如下:  

  1.1、创建一个注解,用来注解切点(pointcut)

package cn.ganlixin.annotation;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface DemoAnnotation {
    //注意这里没有定义属性
}

  

  1.2、创建一个service,使用上面定义的注解来指定切点

  这里为了节约篇幅,就不创建service接口,再创建serviceImpl来实现接口了,直接写在service中:

package cn.ganlixin.service;

import cn.ganlixin.annotation.DemoAnnotation;
import org.springframework.stereotype.Service;

@Service
public class DemoService {

    @DemoAnnotation  // 使用自定义的注解,声明该方法为切点方法
    public void demo() {
        System.out.println("this is DemoService.demo()");
    }
}

  

  1.3、创建Aspect,增加业务逻辑

package cn.ganlixin.aspect;

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class DemoAspect {

    @Before("@annotation(cn.ganlixin.annotation.DemoAnnotation)")
    public void demoBefore() {
        System.out.println("this is before output message");
    }
}

  

  1.4、创建Spring配置类

  主要做的是:指定包扫描路径

package cn.ganlixin;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;

@Configuration
@ComponentScan("cn.ganlixin")
@EnableAspectJAutoProxy
public class AppConfig {
    
}

  

  1.5、测试

package cn.ganlixin;

import cn.ganlixin.service.DemoService;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class AppTest {

    @Test
    public void testAOP1() {
        ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);

        DemoService demoService = context.getBean(DemoService.class);
        demoService.demo();
    }
}

  输出:

this is before output message
this is DemoService.demo()

  

二、获取自定义注解的参数

  2.1、创建带属性的自定义注解

  要获取自定义注解参数,就需要在自定义注解中增加几个属性,下面自定义的TestAnnotation中有两个属性:value和description。

package cn.ganlixin.annotation;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface TestAnnotation {
    String value();
    String description() default "default description";
}

  

  2.2、创建service使用带属性的自定义注解

  service中有两个方法,分别使用了自定义注解:

package cn.ganlixin.service;

import cn.ganlixin.annotation.TestAnnotation;
import org.springframework.stereotype.Service;

@Service
public class TestService {
    
    @TestAnnotation("this is value")
    public void test1() {
        System.out.println("this is TestService.test1()");
    }
    
    @TestAnnotation(value = "this is another value", description = "this is description")
    public void test2() {
        System.out.println("this is TestService.test2()");
    }
}

  

  2.3、创建Aspect的错误示例

  在写博客之前,我也搜过相关的博客,但是发现很多博客中写的都是利用@Around来实现获取注解信息,但是我如果需要在@Before中,@After中获取又怎么办呢?虽然可以通过以下骚操作,通过@Around来模拟@Before和@After,但是还是感觉不好。

  下面还是使用@Before来实现的。

package cn.ganlixin.aspect;

import cn.ganlixin.annotation.TestAnnotation;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class TestAspect {

    @Before("@annotation(cn.ganlixin.annotation.TestAnnotation)")
    public void one(TestAnnotation testAnonotation) {
        System.out.println(testAnonotation.value());
        System.out.println(testAnonotation.description());
    }
}

  上面的代码看似没有问题,one()方法中接收一个TestAnnotation的参数,以为能够获取到切点方法的注解信息,但是,IDE会告诉你如下错误:

  

 

  2.4、创建Aspect的正确做法

package cn.ganlixin.aspect;

import cn.ganlixin.annotation.TestAnnotation;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class TestAspect {

    // 先创建一个方法,方法名随意,但是需要制定@annotation为刚刚自定义的注解
    @Pointcut("@annotation(cn.ganlixin.annotation.TestAnnotation)")
    public void test() {}

    // 使用@Before,需要先引入上面@Pointcut注解的方法名,在加上@annotation,
    // @annotation中的值,需要和action方法中的参数名称相同(必须相同,但是名称任意)
    @Before("test() && @annotation(testAnnotation)")
    public void action(TestAnnotation testAnnotation) {
        System.out.println("Annotation value : " + testAnnotation.value());
        System.out.println("Annotation description : " + testAnnotation.description());
        System.out.println("this is TestAspect.action()");
    }
}

  划重点

// 第2个示例,强调@annotation中的值,需要和方法参数名相同
@Before("test() && @annotation(abcdef)")
public void action2(TestAnnotation abcdef) {
    System.out.println("Annotation value : " + abcdef.value());
    System.out.println("Annotation description : " + abcdef.description());
    System.out.println("this is TestAspect.action()");
}

  

   2.5、测试

  Spring的配置类不用更改,测试代码如下:

package cn.ganlixin;

import cn.ganlixin.service.TestService;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class AppTest {
    @Test
    public void testAOP2() {
        ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
        TestService testService = context.getBean(TestService.class);
        
        testService.test1();
        System.out.println("----------------------------");
        testService.test2();
    }
}

  输出:

Annotation value : this is value
Annotation description : default description
this is TestAspect.action()
this is TestService.test1()
----------------------------
Annotation value : this is another value
Annotation description : this is description
this is TestAspect.action()
this is TestService.test2()

  

三、总结

  要想是获取AOP中自定义注解的参数值,主要就一点:

  

 

转载于:https://www.cnblogs.com/-beyond/p/11387487.html

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
获取自定义注解修饰的Controller参数值,可以使用 AOP方法拦截请求,并在拦截方法获取参数值。具体步骤如下: 1. 定义一个自定义注解,例如 @MyAnnotation。 2. 在需要拦截的 Controller 方法参数上添加 @MyAnnotation 注解。 3. 编写一个切面类,在类上添加 @Aspect 注解,并定义一个切点方法,使用 @Around 注解指定需要拦截的方法。 4. 在切点方法,使用 JoinPoint 对象获取目标方法的参数,通过反射获取参数上的 @MyAnnotation 注解,从而获取注解的值。 示例代码如下: ```java @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.PARAMETER) public @interface MyAnnotation { String value(); } @Controller public class UserController { @GetMapping("/users/{id}") public String getUser(@MyAnnotation("user_id") @PathVariable Long id) { // ... } } @Aspect @Component public class MyAspect { @Around("execution(* com.example.controller.*.*(..))") public Object around(ProceedingJoinPoint point) throws Throwable { Object[] args = point.getArgs(); for (Object arg : args) { MyAnnotation annotation = arg.getClass().getAnnotation(MyAnnotation.class); if (annotation != null) { String value = annotation.value(); // TODO: 处理注解值 } } return point.proceed(); } } ``` 在上面的示例,我们定义了一个 @MyAnnotation 注解,用于修饰 Controller 方法的参数。在 UserController ,我们使用 @MyAnnotation 注解修饰了 getUser 方法的 id 参数,注解的值为 "user_id"。 然后,在 MyAspect 类,我们使用 @Around 注解指定需要拦截的方法,并通过 JoinPoint 对象获取目标方法的参数。我们遍历参数列表,使用反射获取参数上的 @MyAnnotation 注解,并从注解获取值,最终可以进行处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值