有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

Spring(Java):在非线性类层次结构中不会触发方面

当类层次结构不是线性的时,在基接口上定义时不会触发方面

最有趣的是:当向实现的父类添加委托实现(参见最后一个代码块)时,测试变为绿色(方面按预期触发)

问题:为什么它不能像示例中描述的那样工作,为什么它能与授权实现一起工作

示例(抱歉,找不到较短的示例):

测试:

 @Autowired
private TheInterface underTest;
private static boolean aspectCalled;
private static boolean implementationCalled;

@Test
public void aspectTest() throws Exception {
    aspectCalled = false;
    implementationCalled = false;

    underTest.doSomething();

    assertTrue("Implementation not called!", implementationCalled);
    assertTrue("Aspect not called!", aspectCalled);
}

方面:

@Aspect
@Component
public static class MyAspect {

    @Before("execution(* *..SpecializedInterface+.doSomething())")
    public void applyAspect() {
        aspectCalled = true;
    }
}

接口:

public static interface TheInterface {
    void doSomething();
}

public static interface SpecializedInterface extends TheInterface {
    // inherits doSomething
    // defines some other methods
}

抽象实现(模板模式):

public static abstract class BaseTemplate implements TheInterface {
    abstract void doOneStep();

    @Override
    public void doSomething() {
        // do some stuff and
        doOneStep();
    }
}

public static abstract class SpecializedTemplate extends BaseTemplate implements SpecializedInterface {
    // some other methods
}

实现bean:

@Component
public static class TemplateImplementation extends SpecializedTemplate {
    @Override
    void doOneStep() {
        implementationCalled = true;
    }
}

(如果您感兴趣:测试设置:)

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = MyConfig.class)
public class AopTest {
    @Configuration
    @EnableAspectJAutoProxy
    @ComponentScan(basePackageClasses = AopTest.class)
    public static class MyConfig {
    }
    ...

难看的解决方法:将此片段添加到SpecializedTemplate

    @Override
    public void doSomething() {
        super.doSomething();
    }

那么,为什么需要这种变通方法呢


共 (1) 个答案

  1. # 1 楼答案

    Thomas Stets已经解释了字节码和JVM的内容,所以我将为您的问题提供一个解决方案,另请参见我对一个非常类似的问题的回答

    @Aspect
    public static class MyAspect {
        @Before("execution(* *..TheInterface+.doSomething()) && target(specializedInterface)")
        public void applyAspect(SpecializedInterface specializedInterface) {
            aspectCalled = true;
        }
    }
    

    也就是说,切入点的目标是实际定义方法的基本接口,然后将目标限制在您选择的专用子接口上。这会让你的测试变成绿色