gpt4 book ai didi

java - 使用aspectj进行springboot日志记录获取IllegalArgumentException:错误在::0在切入点中正式未绑定(bind)

转载 作者:行者123 更新时间:2023-12-01 09:06:54 24 4
gpt4 key购买 nike

我想在 springboot 项目中创建一个aspectJ组件,只要@Loggable注释存在,方法或类,或者两者都存在(将考虑方法),它就会打印日志消息。

可记录注释:

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.TYPE, ElementType.METHOD })
public @interface Loggable {
boolean duration() default false;
}

Aspectj 类:

@Aspect
@Component
public class LogInterceptorAspect {

@Pointcut("execution(public * ((@Loggable *)+).*(..)) && within(@Loggable *)")
public boolean loggableDefinition(Loggable loggable) {
return loggable.duration();
}

@Around("loggableDefinition(withDuration)")
public void log(ProceedingJoinPoint joinPoint, boolean withDuration) throws Throwable {
getLogger(joinPoint).info("start {}", joinPoint.getSignature().getName());

StopWatch sw = new StopWatch();
Object returnVal = null;
try {
sw.start();
returnVal = joinPoint.proceed();
} finally {
sw.stop();
}

getLogger(joinPoint).info("return value: {}, duration: {}", returnVal, sw.getTotalTimeMillis()));

}

private Logger getLogger(JoinPoint joinPoint) {
return LoggerFactory.getLogger(joinPoint.getSignature().getDeclaringType());
}
}

通过上面的代码我得到

java.lang.IllegalArgumentException: error at ::0 formal unbound in pointcut

出了什么问题?

最佳答案

基本上,形式参数在切入点上是未绑定(bind)的。

这是一个基于本文详细介绍的方法的替代工作示例:@AspectJ Class level Annotation Advice with Annotation as method argument

出于以下几个原因,我稍微修改了您的方法以避免出现问题:

  • 简化了最初的切入点并赋予其单一职责

    • 给它一个描述性的名称来表明其用途
    • 通过消除对 Loggable 的依赖使其更具可重用性
    • 使其实现与大多数可用示例文档保持接近
  • 将建议分为两个更简单的方法,每个方法都有一个易于理解的职责

    • 通过删除花哨的运算符简化了表达式
    • 将注释直接注入(inject)到使用的 Advice 中,而不是尝试从 PointCut 传递,这感觉像是不必要的复杂性
    • 使其实现与大多数可用示例文档保持接近
  • 添加了单元测试的开始以验证预期行为,以便可以负责任地对 PointCut 和 Advice 表达式进行更改(您应该完成它)

在使用切入点/建议表达式时,我通常会尝试尽可能寻找最简单、最清晰的解决方案,并对它们进行彻底的单元测试,以确保得到我期望的行为。下一个查看您的代码的人将会欣赏它。

希望这有帮助。

package com.spring.aspects;

import static org.junit.Assert.assertEquals;

import org.aspectj.lang.JoinPoint;
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.junit.Test;
import org.junit.runner.RunWith;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.util.StopWatch;

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

@RunWith(SpringRunner.class)
@ContextConfiguration(classes = AspectInjectAnnotationTest.TestContext.class)
public class AspectInjectAnnotationTest {

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.TYPE, ElementType.METHOD })
public @interface Loggable {
boolean duration() default false;
}

@Aspect
public static class LogInterceptorAspect {

@Pointcut("execution(public * *(..))")
public void anyPublicMethod() {
}

@Around("anyPublicMethod() && @annotation(loggable)")
public Object aroundLoggableMethods(ProceedingJoinPoint joinPoint, Loggable loggable) throws Throwable {
return log(joinPoint, loggable);
}

@Around("(anyPublicMethod() && !@annotation(AspectInjectAnnotationTest.Loggable)) && @within(loggable)")
public Object aroundPublicMethodsOnLoggableClasses(ProceedingJoinPoint joinPoint, Loggable loggable)
throws Throwable {
return log(joinPoint, loggable);
}

public Object log(ProceedingJoinPoint joinPoint, Loggable loggable) throws Throwable {
getLogger(joinPoint).info("start [{}], duration [{}]", joinPoint.getSignature().getName(),
loggable.duration());

StopWatch sw = new StopWatch();
Object returnVal = null;
try {
sw.start();
returnVal = joinPoint.proceed();
} finally {
sw.stop();
}

getLogger(joinPoint).info("return value: [{}], duration: [{}]", returnVal, sw.getTotalTimeMillis());

return returnVal;
}

private Logger getLogger(JoinPoint joinPoint) {
return LoggerFactory.getLogger(joinPoint.getSignature().getDeclaringType());
}
}

// class level annotation - should only proxy public methods
@Loggable(duration = true)
public static class Service1 {

// public - should be proxied
public String testS1M1(String test) {
return testProtectedM(test);
}

// public - should be proxied
public String testS1M2(String test) {
return testProtectedM(test);
}

// protected - should not be proxied
protected String testProtectedM(String test) {
return testPrivateM(test);
}

// private - should not be proxied
private String testPrivateM(String test) {
return test;
}
}

// no annotation - class uses method level
public static class Service2 {

@Loggable
public String testS2M1(String test) {
return protectedMethod(test);
}

// no annotation - should not be proxied
public String testS2M2(String test) {
return protectedMethod(test);
}

// protected - should not be proxied
protected String protectedMethod(String test) {
return testPrivate(test);
}

// private - should not be proxied
private String testPrivate(String test) {
return test;
}
}

// annotation - class and method level - make sure only call once
@Loggable
public static class Service3 {

@Loggable
public String testS3M1(String test) {
return test;
}
}

// context configuration for the test class
@Configuration
@EnableAspectJAutoProxy
public static class TestContext {

// configure the aspect
@Bean
public LogInterceptorAspect loggingAspect() {
return new LogInterceptorAspect();
}

// configure a proxied beans
@Bean
public Service1 service1() {
return new Service1();
}

// configure a proxied bean
@Bean
public Service2 service2() {
return new Service2();
}

// configure a proxied bean
@Bean
public Service3 service3() {
return new Service3();
}
}

@Autowired
private Service1 service1;

@Autowired
private Service2 service2;

@Autowired
private Service3 service3;

@Test
public void aspectShouldLogAsExpected() {
// observe the output in the log, but craft this into specific
// unit tests to assert the behavior you are expecting.

assertEquals("service-1-method-1", service1.testS1M1("service-1-method-1")); // expect logging
assertEquals("service-1-method-2", service1.testS1M2("service-1-method-2")); // expect logging
assertEquals("service-2-method-1", service2.testS2M1("service-2-method-1")); // expect logging
assertEquals("service-2-method-2", service2.testS2M2("service-2-method-2")); // expect no logging
assertEquals("service-3-method-1", service3.testS3M1("service-3-method-1")); // expect logging once


}
}

关于java - 使用aspectj进行springboot日志记录获取IllegalArgumentException:错误在::0在切入点中正式未绑定(bind),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41197432/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com