AOP Spring @AfterReturning не работает должным образом

Я изучаю AOP spring и пробую несколько примеров. Что касается @AfterReturning, я понимаю, что метод вызывается только в том случае, если цель успешно возвращена и соответствует pointcut. Однако в моем случае, как показано ниже, у меня есть pointcut, который определяет все методы, которые возвращают только строку, но вызывает все методы void, а также метод, который возвращает строку.

Мой совет:

@AfterReturning(value= "execution(*  com.aop..CustomerServiceImpl.*(..))",
        returning= "string")
public void returnStringPointcut(JoinPoint joinPoint,String string){

    System.out.println("logAfter() is running!");
    System.out.println("String : " + string);
    System.out.println("hijacked : " + joinPoint.getSignature().getName());
    System.out.println("******");

}

Пожалуйста, найдите ниже мой класс Impl ниже:

public void addCustomer() {
    // TODO Auto-generated method stub

}

public String getCustomer() {
    // TODO Auto-generated method stub
    return "string";
}

public boolean deleteCustomer() {
    // TODO Auto-generated method stub
    return false;
}
public void addCustomerAppended() {
    // TODO Auto-generated method stub

}

public void deleteCustomerVoid() {
    // TODO Auto-generated method stub
    //return false;
}

Пожалуйста, найдите ниже мой класс MainApp:

public class App {
    public static void main(String[] args)

    {
        ApplicationContext context = new ClassPathXmlApplicationContext(
                "Spring-Customer.xml");

        CustomerService cs =  context.getBean("customerBo", CustomerService.class);
        cs.addCustomer();
        cs.addCustomerAppended();
        cs.deleteCustomer();
        cs.deleteCustomerVoid();
        cs.getCustomer();


    }
}

Я ожидал, что будет вызываться только getCustomer(), поскольку он единственный, который возвращает строку, но вместо этого я получаю следующий вывод в своей консоли при запуске приложения:

logAfter() is running!
String : null
hijacked : addCustomer
******
logAfter() is running!
String : null
hijacked : addCustomerAppended
******
logAfter() is running!
String : null
hijacked : deleteCustomerVoid
******
logAfter() is running!
String : string
hijacked : getCustomer
******

Пожалуйста, найдите мой pom.xml ниже:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.aop.spring</groupId>
    <artifactId>SpringAopOnly</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>SpringAopOnly</name>
    <url>http://maven.apache.org</url>
    <properties>
        <spring.version>3.0.5.RELEASE</spring.version>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    </properties>

    <dependencies>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.8.2</version>
            <scope>test</scope>
        </dependency>

        <!-- Spring 3 dependencies -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <!-- Spring AOP  -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aop</artifactId>
            <version>${spring.version}</version>
        </dependency>

            <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjrt</artifactId>
            <version>1.6.11</version>
        </dependency>

        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.6.11</version>
            </dependency>

    </dependencies>
</project>

Пожалуйста, найдите мой файл конфигурации ниже:

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
    http://www.springframework.org/schema/beans/spring-beans-3.0.xsd 
    http://www.springframework.org/schema/aop 
    http://www.springframework.org/schema/aop/spring-aop-3.0.xsd ">

    <!-- <aop:aspectj-autoproxy />  -->
    <aop:aspectj-autoproxy>
    <aop:include name ="logAspect" />
    </aop:aspectj-autoproxy>

    <bean id="customerBo" class="com.aop.impl.CustomerServiceImpl" />

    <!-- Aspect -->
    <bean id="logAspect" class="com.aop.aspect.CustomerAspect" />

</beans>

Любая идея, почему он вызывает метод void, пожалуйста?

Я также попытался изменить afterReturning со String на boolean, тогда я получаю ожидаемый результат, который вызывается только deleteCustomer, поскольку он возвращает логическое значение.

Заранее большое спасибо за ответ.


person user1999453    schedule 22.12.2013    source источник
comment
Я не могу воспроизвести то, что вы видите. Есть что-то, что вы нам не показываете? Какую версию Spring и AspectJ вы используете?   -  person Sotirios Delimanolis    schedule 22.12.2013
comment
Большое спасибо за ответ, я обновил свои вопросы   -  person user1999453    schedule 22.12.2013


Ответы (1)


Аргумент returning указывает только

Имя аргумента в сигнатуре совета для привязки возвращаемого значения к

Ваш фактический pointcut

@AfterReturning(value= "execution(*  com.aop..CustomerServiceImpl.*(..))",
    returning= "string")

указывает

execution(* com.aop..CustomerServiceImpl.*(..))

где * соответствует всем возвращаемым типам.

Вы должны изменить его на

execution(java.lang.String com.aop..CustomerServiceImpl.*(..))

если вам нужны только методы, объявленные как возвращающие String.


И тип возвращаемого значения в выражении value аннотации, и тип параметра метода играют роль в том, какие методы будут рекомендованы. Например

@AfterReturning(value = "execution(String com.aop..CustomerServiceImpl.*(..))", returning = "random")
public void returnStringPointcut(JoinPoint joinPoint, Integer random) {

ни с чем не совпадет.


Кстати, вам следует подумать об обновлении версий Spring и аспекта. Я думаю, что многие из этих проблем либо исправлены, либо библиотеки в целом стали более стабильными.

person Sotirios Delimanolis    schedule 22.12.2013