1

Spring Framework在 Web 环境中使用 4.3.3。

我有一个@Controller用于Web请求的方法Web Browser,它使用其他方式的依赖关系,@Controller但用于Rest目的。后者提到使用@Service等...

这种关于使用“Rest”的“Web”的方法如何在使用 Spring MVC的内容协商中解释依赖关系Combining Data and Presentation Formats。直到这里进行开发/测试和生产工作正常。这是一种有价值的方法。

注意该类Rest带有注释,@Controller因为我使用ResponseEntity<?>@ResponseBody

问题在于AOP

关于它的基础设施,我有:

@Configuration
@EnableAspectJAutoProxy
public class AopConfig {

}

关于@Controllers 我有这两个类:

  • PersonaDeleteOneController和:
    • deleteOne(@PathVariable String id, Model model)为了@GetMapping
    • deleteOne(@PathVariable String id, RedirectAttributes redirectAttributes)为了@DeleteMapping
  • PersonaRestController
    • deleteOne(@PathVariable String id)为了@DeleteMapping

这两个类在同一个包中声明,名为:

  • com.manuel.jordan.controller.persona

我有以下内容@Pointcut

@Pointcut(value=
"execution(* com.manuel.jordan.controller.*.*Controller.deleteOne(String, ..)) 
&& args(id) && target(object)")
public void deleteOnePointcut(String id, Object object){}

@Pointcut用于以下建议:

@Before(value="ControllerPointcut.deleteOnePointcut(id, object)")
public void beforeAdviceDeleteOne(String id, Object object){
    logger.info("beforeAdviceDeleteOne - @Controller: {} - Method: deleteOne - id: {}", object.getClass().getSimpleName(), id);
}

当我执行Rest测试时,我可以通过AOP + logging打印以下模式来确认:

  • @Controller休息)-> @Service->@Repository

直到这里所有的工作都按预期进行

当我执行Web测试时,我可以通过AOP + logging打印以下模式来确认:

  • @Controller休息)-> @Service->@Repository

我需要或期望的是以下内容:

  • @Controller网络)-> @Controller休息)-> @Service->@Repository

有什么问题或遗漏?deleteOne签名对它们的参数没有歧义。

生产的情况相同。

Α

这里的控制器:

@Controller
@RequestMapping(value="/personas")
public class PersonaDeleteOneController {

    private final PersonaRestController personaRestController;

    @Autowired
    public PersonaDeleteOneController(PersonaRestController personaRestController){
        this.personaRestController = personaRestController;
    }

    @GetMapping(value="/delete/{id}",
                produces=MediaType.TEXT_HTML_VALUE)
    public String deleteOne(@PathVariable String id, Model model){
        model.addAttribute(personaRestController.findOneById(id));
        model.addAttribute("root", "/personas/delete");
        return "persona/deleteOne";
    }

    @DeleteMapping(value="/delete/{id}",
                   produces=MediaType.TEXT_HTML_VALUE)
    public String deleteOne(@PathVariable String id, RedirectAttributes redirectAttributes){
        personaRestController.deleteOne(id);
        redirectAttributes.addFlashAttribute("message", "process.successful");
        return "redirect:/message";
    }

}

@Controller
@RequestMapping(value="/personas")
public class PersonaRestController {

    private final PersonaService personaService;

    @Autowired
    public PersonaRestController(PersonaService personaService){
        this.personaService = personaService;
    }

    @DeleteMapping(value="/{id}")
    public ResponseEntity<Void> deleteOne(@PathVariable String id){
        personaService.deleteOne(id);
        return ResponseEntity.noContent().build();
    }

    ....

你怎么能看到我不使用this.来执行方法调用。

4

1 回答 1

1

似乎问题出在您的pointcut定义中。您可能会注意到您的通知方法仅对具有一个参数的方法执行,因此这是由于您args(id)在切入点声明中指定的事实。如果您 remove args(id),它必须按预期工作,但在这种情况下,必须使用一些变通方法来公开参数值。

我认为这是 AspectJ 的奇怪行为,因为像这样的结构execution(* *.*(String, ..)) && args(arg) && target(t))具有清晰的语义,可以捕获所有带有String第一个参数的方法并将其暴露给args. 至少对于 AspectJ 开发人员来说,它可能是一个错误或特性。

要获得您想要的,您可以使用joinPoint.getArgs()内部建议方法的解决方法,如下所示:

@Pointcut(value=
"execution(* com.manuel.jordan.controller.*.*Controller.deleteOne(..)) && target(object)")
public void deleteOnePointcut(Object object){}

@Before(value="ControllerPointcut.deleteOnePointcut(object)")
public void beforeAdviceDeleteOne(JoinPoint jp, Object object){
    Object id = jp.getArgs()[0];
    logger.info("beforeAdviceDeleteOne - @Controller: {} - Method: deleteOne - id: {}", object.getClass().getSimpleName(), id);
}
于 2016-09-29T03:51:46.107 回答