Spring AOP не применяется к методам с возвращаемыми значениями типа Generi c - PullRequest
0 голосов
/ 01 марта 2020

РЕДАКТИРОВАТЬ: Решено, потому что вначале из-за двойных звездочек я скопировал его из Книги: Spring Microservices in Action.

Я пытаюсь посоветовать некоторые методы контроллера .

Это класс аспектов:

@Aspect
public class PBLogger {

    @Pointcut("execution(** com.sunwell.product..*.*(..))")
    public void standardMethod() {
    }

    @Around("standardMethod()")
    public Object log(ProceedingJoinPoint jp) {
           ...
    }

...
}

Это целевой метод, я убедился, что имя пакета правильное.

Этот не соответствует работа (метод извещения не вызывается):

@RequestMapping(value = "resources/items", method = RequestMethod.GET,
                    produces = "application/json"
            )
public ResponseEntity<Map<String,Object>> getItems(
                    @RequestHeader(value="Authorization", required = false) String _auth, @RequestParam(value="systemId", required = false) Integer _i ) throws Exception 
{ ... }

Без универсального c типа возвращаемого значения он работает:

@RequestMapping(value = "resources/items", method = RequestMethod.GET,
                        produces = "application/json"
                )
public void getItems(@RequestHeader(value="Authorization", required = false) String _auth, @RequestParam(value="systemId", required = false) Integer _i ) throws Exception 
{ ... }

Он также работает с использованием аннотированного pointcut вместо выполнения

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Example
{
    public String msg();
}

@Aspect
public class PBLogger {


        @Pointcut("@annotation(Example)")
        public void standardMethod() {
        }

        @Around("standardMethod()")
        public Object log(ProceedingJoinPoint jp) {
           ...
        }

    ...
}

//target method:
        @RequestMapping(value = "resources/items", method = RequestMethod.GET,
                    produces = "application/json"
            )
        @Example
public ResponseEntity<Map<String,Object>> getItems(
                    @RequestHeader(value="Authorization", required = false) String _auth, @RequestParam(value="systemId", required = false) Integer _ ) throws Exception 
{ ... }

1 Ответ

2 голосов
/ 02 марта 2020

Подпись целевого метода не является проблемой, обобщенной или нет. Посмотрите на мои MCVE :

Целевые классы:

Обратите внимание, что вы не можете поместить оба метода в один и тот же класс, потому что они будут иметь одинаковые Тип стирания. Компилятор будет жаловаться на это.

package de.scrum_master.spring.q60474362;

import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.annotation.RequestHeader;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;

import java.util.Map;

@Component
public class MyTargetClass {
  @RequestMapping(
    value = "resources/items", method = RequestMethod.GET,
    produces = "application/json"
  )
  public ResponseEntity<Map<String, Object>> getItems(
    @RequestHeader(value = "Authorization", required = false) String _auth,
    @RequestParam(value = "systemId", required = false) Integer _i
  ) throws Exception
  {
    System.out.println("MyTargetClass.getItems");
    return null;
  }
}
package de.scrum_master.spring.q60474362;

import org.springframework.stereotype.Component;
import org.springframework.web.bind.annotation.RequestHeader;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;

@Component
public class MyOtherTargetClass {
  @RequestMapping(
    value = "resources/items", method = RequestMethod.GET,
    produces = "application/json"
  )
  public void getItems(
    @RequestHeader(value = "Authorization", required = false) String _auth,
    @RequestParam(value = "systemId", required = false) Integer _i
  ) throws Exception
  {
    System.out.println("MyOtherTargetClass.getItems");
  }
}

Аспект:

Обратите внимание, что ** в вашем примере pointcut не имеет смысла, я изменил это с использованием *. Если ** работает, вам просто повезло, что проверка синтаксиса не является строгой, но моя IDE на самом деле жалуется, как и должно. Кроме того, вам на самом деле не нужно my.package..*.*(..), достаточно my.package..*(..), если не имеют значения и имя класса, и метод.

package de.scrum_master.spring.q60474362;

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

@Aspect
@Component
public class PBLogger {
  @Pointcut("execution(* de.scrum_master.spring..*(..))")
  public void standardMethod() {}

  @Around("standardMethod()")
  public Object log(ProceedingJoinPoint jp) throws Throwable {
    System.out.println(jp);
    return jp.proceed();
  }
}

Приложение и конфигурация драйвера Spring:

package de.scrum_master.spring.q60474362;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;

@SpringBootApplication
@Configuration
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class Application {
  public static void main(String[] args) throws Exception {
    try (ConfigurableApplicationContext appContext = SpringApplication.run(Application.class, args)) {
      doStuff(appContext);
    }
  }

  private static void doStuff(ConfigurableApplicationContext appContext) throws Exception {
    MyTargetClass myTargetClass = appContext.getBean(MyTargetClass.class);
    myTargetClass.getItems("x", 11);
    MyOtherTargetClass myOtherTargetClass = appContext.getBean(MyOtherTargetClass.class);
    myOtherTargetClass.getItems("y", 22);
  }
}

Журнал консоли:

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v1.5.2.RELEASE)

(...)
2020-03-02 09:00:49.712  INFO 24028 --- [           main] d.s.spring.q60474362.Application         : Started Application in 3.554 seconds (JVM running for 5.157)
execution(ResponseEntity de.scrum_master.spring.q60474362.MyTargetClass.getItems(String,Integer))
MyTargetClass.getItems
execution(void de.scrum_master.spring.q60474362.MyOtherTargetClass.getItems(String,Integer))
MyOtherTargetClass.getItems
(...)

Если он не работает для вас, у вас есть другая проблема, например, проблема с целевым классом c не выполняется. обнаруживается при сканировании компонента, например, из-за забытой аннотации @Component или чего-либо еще.

...