简体   繁体   English

Spring boot 自定义注解中的方面

[英]Aspect in Spring boot Custom Annotation

I am creating a custom annotation NullCheck for method parameter to check value is null or not hello(@NullCheck String text) , but I am not able to invoke Aspect around the Annotation.我正在为方法参数创建自定义注释 NullCheck 以检查值是否为空hello(@NullCheck String text) ,但我无法围绕注释调用 Aspect。

Main Class主类

package com.example.demo;
import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.context.annotation.EnableAspectJAutoProxy;
    @SpringBootApplication
    @EnableAutoConfiguration
    @EnableAspectJAutoProxy
    public class DemoApplication {
        public static void main(String[] args) {
            SpringApplication.run(DemoApplication.class, args);
        }
    }

Controller class, just invoking an aspect for POC, not returning anything控制器类,只为 POC 调用一个方面,不返回任何东西

package com.example.demo;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController; 
@RestController
@RequestMapping("/")
class HelloController {
    @GetMapping
    public void hello() {
        hello("hi");
    }
    private void hello(@NullCheck String text) {
        System.out.println(text);
    }
}

Annotation注解

package com.example.demo;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
@Documented
@Retention(RUNTIME)
@Target(ElementType.PARAMETER)
public @interface NullCheck { }

Aspect方面

package com.example.demo;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class NullCheckAspect {

this is working这是工作

@Before("@annotation(org.springframework.web.bind.annotation.GetMapping)")

but this is not但这不是

@Before("@annotation(com.example.demo.NullCheck)")
    public void beforeAdvice(JoinPoint joinPoint) {
        System.out.println("Before method:" + joinPoint.getSignature());
    }
}

build.gradle构建.gradle

buildscript {
    ext {
        springBootVersion = '2.1.2.RELEASE'
    }
    repositories {
        mavenCentral()
    }
    dependencies {
        classpath("org.springframework.boot:spring-boot-gradle-plugin:${springBootVersion}")
    }
}
apply plugin: 'java'
apply plugin: 'idea'
apply plugin: 'eclipse'
apply plugin: 'org.springframework.boot'
apply plugin: 'io.spring.dependency-management'
group = 'com.example'
version = '0.0.1-SNAPSHOT'
sourceCompatibility = '1.8'
idea {
    module {
        // if you hate browsing Javadoc
        downloadJavadoc = true
        // and love reading sources :)
        downloadSources = true
    }
}
bootJar {
    launchScript()
}
repositories {
    mavenCentral()
    jcenter()
}
bootJar {
    launchScript()
}
dependencies {
    implementation 'org.springframework.boot:spring-boot-starter-actuator'
    implementation 'org.springframework.boot:spring-boot-starter-aop'
    implementation 'org.springframework.boot:spring-boot-starter-web'
    runtimeOnly 'org.springframework.boot:spring-boot-devtools'
}

what am I missing?我错过了什么?

As per my understanding and after doing some search on google, you can get method parameter and its value with particular annotation with following code:根据我的理解,在谷歌上进行一些搜索后,您可以使用以下代码获取带有特定注释的方法参数及其值:

package com.example.demo;

import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class NullCheckAspect {

    @Around("execution(* com.example.demo.HelloController.nullChecker(String))")
    public Object around(ProceedingJoinPoint pJoinPoint) throws Throwable {

        Object[] args = pJoinPoint.getArgs();

        Method method = MethodSignature.class.cast(pJoinPoint.getSignature()).getMethod();

        Annotation[][] parametersAnnotations = method.getParameterAnnotations();

        Map<String, Object> annotatedParameters = new HashMap<>();

        int i = 0;
        for(Annotation[] parameters : parametersAnnotations) {
                Object arg = args[i];
                String name = method.getParameters()[i++].getDeclaringExecutable().getName();
                for(Annotation parameter: parameters) {
                    if ((parameter instanceof NullCheck)) {
                        System.out.println("Found the null checker annotation with name: " + name);
                        System.out.println("Found the null checker annotation with arg:  " + arg);
                        annotatedParameters.put(name, arg);
                }
                }
        }
        System.out.println(annotatedParameters);

        return pJoinPoint.proceed(args);
    }
}

And with the interface annotation:并带有接口注释:

package com.example.demo;

import static java.lang.annotation.RetentionPolicy.RUNTIME;

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

import org.springframework.stereotype.Component;

@Component
@Retention(RUNTIME)
@Target(ElementType.PARAMETER)
public @interface NullCheck {
}

More details about my code you can check it at my github repository, I have created spring boot demo version and pushed it for you at https://github.com/krishnaiitd/learningJava/blob/master/springBoot/gs-spring-boot/src/main/java/com/example/demo/HelloController.java有关我的代码的更多详细信息,您可以在我的 github 存储库中查看它,我已经创建了 Spring Boot 演示版本并在https://github.com/krishnaitd/learningJava/blob/master/springBoot/gs-spring-boot 上为您推送/src/main/java/com/example/demo/HelloController.java

This also include other type of aspects like tracking the time of a particular methods.这还包括其他类型的方面,例如跟踪特定方法的时间。

Hope this will help you to get the basic understanding of @Aspect in Spring boot.希望这能帮助您对 Spring boot 中的@Aspect有基本的了解。

I don't know why it is as it is but I had the same issue.我不知道为什么会这样,但我遇到了同样的问题。 This is can be solved easily by using Pointcuts, for example:这可以通过使用切入点轻松解决,例如:

@Before("nulllCheckAnnotation()")
    public void beforeAdvice(JoinPoint joinPoint) {
        System.out.println("Before method:" + joinPoint.getSignature());
    }
}

@Pointcut("@annotation(com.example.demo.NullCheck)")
private void nulllCheckAnnotation() { }

Read more about pointcuts here if you interested: https://www.baeldung.com/spring-aop-pointcut-tutorial如果您有兴趣,在此处阅读有关切入点的更多信息: https : //www.baeldung.com/spring-aop-pointcut-tutorial

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM