8

我用原始的切入点和建议方法编写了简单的方面:

@Aspect
public class MyAspect {

  @Pointcut("execution(static * com.mtag.util.SomeUtil.someMethod(..))")
  public void someMethodInvoke() { }

  @AfterReturning(value = "someMethodInvoke())", returning = "comparisonResult")
  public void decrementProductCount(List<String> comparisonResult) {
    //some actions
  }
}

我有以下基于 Spring 注释的应用程序配置:

@Configuration
@EnableAspectJAutoProxy
public class AppConfig { 
  //...
}

以及 com.mtag.util 包中的实用程序类:

public class SomeUtil {
  static List<String> someMethod(List<String> oldList, List<String> newList) {
    //...
  } 
}

但是当我打电话时

SomeUtil.someMethod(arg1, arg2);

在单元测试中,我可以看到方法调用没有被拦截,并且我的 @AfterReturning 建议不起作用。

但是,如果我将 someMethod() 类型更改为实例(非静态)方法,则切入点

@Pointcut("execution(* com.mtag.util.SomeUtil.someMethod(..))")

并通过添加 @Component 注释和调用目标方法来管理 SomeUtil bean,如下所示:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = {AppConfig.class}, loader = AnnotationConfigContextLoader.class)
public class SomeUtilTest {

    @Autowired
    private SomeUtil someUtil;

    @Test
    public void categoriesDiffCalc() {
        List<String> result = someUtil.someMethod(...);
    }
}

一切都好。

我可以通过什么方式为静态方法设置建议?

4

1 回答 1

8

实际上在spring框架中没有使用自动代理拦截静态方法的解决方案。您应该使用 LWT AspectJ 解决方案。

简而言之,您应该使用相同的注释,但需要一些额外的配置。

1)添加到spring上下文文件的下一行:

<context:load-time-weaver/>

(可能在你的情况下没有必要)

2) 不幸的是,您还应该添加 META-INF/aop.xml。例子:

<weaver>
    <include within="com.example.ClassA"/> <!-- path to concrete class -->
    <include within="com.log.* "/> <!—- path to aspects package -->
</weaver>
<aspects>
    <aspect name="com.log.AspectA"/>
</aspects>

3) 启动 JVM 时的参数

-javaagent:${PATH_TO_LIB }/aspectjweaver.jar

应该添加。

所以这个解决方案相当费力。

有关更多信息,请阅读此处的第 7.8.4 章http://docs.spring.io/spring/docs/3.0.0.RC2/reference/html/ch07s08.html

于 2014-09-11T07:44:46.303 回答