将AOP功能添加到Spring项目后,没有发现合格的Bean错误

时间:2019-07-14 06:07:51

标签: java spring aspectj

在将AOP功能添加到我的简单项目中时出现以下错误,有人可以帮我了解一下吗? 我还在下面提到了代码的相关部分。

Exception in thread "main" org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'com.AOP.Car' available
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:346)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:337)
    at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:1123)
    at com.AOP.App.main(App.java:13)

package com.AOP;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;    
@Configuration
@ComponentScan(basePackages = "com.AOP")
public class AppConfig {        
}

package com.AOP;

import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class App 
{
    public static void main( String[] args )
    {
        ApplicationContext context = new  AnnotationConfigApplicationContext(AppConfig.class);
        Car car =  context.getBean(Car.class);      
        car.drive();
    }
}

package com.AOP;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Component
public class Car implements Vehicle{

    @Autowired
    private Tyre tyre;      
    public Tyre getTyre() {
        return tyre;
    }

    public void setTyre(Tyre tyre) {
        this.tyre = tyre;
    }

    public void drive()
    {
        System.out.println("driving a car");
        System.out.println(tyre);
    }
}

package com.AOP;

public interface Vehicle {
    void drive();

}

package com.AOP;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;
@Component
@Aspect
@EnableAspectJAutoProxy
public class AOPClass {

    @After("execution(* setT*(..))")    
    public void afterAdvice2()
    {
        System.out.println("After Advise (set) start here");
    }

    @Before("execution(* drive(..))")   
    public void beforeAdvice1()
    {
        System.out.println("Before Advise (drive) start here");
    }
}

如果我得到一个简单的类“ Car”而不实现“ Vehicle”接口,则一切正常。但是添加该扩展名将导致menterror。

1 个答案:

答案 0 :(得分:1)

Spring documentation中对此行为进行了很好的描述。 您的情况就是这样:

  

如果要代理的目标对象至少实现一个接口,则使用JDK动态代理。

您仍然可以通过名称获取Bean,并查看其类是什么:

    Object b = context.getBean("car");      
    System.out.println(b.getClass().getName());

类似于com.sun.proxy.$Proxy38,如果您尝试浏览其界面,则其中会有com.AOP.Vehicle

在这一点上,很明显为什么无法按类获取bean。

该怎么办?有一些选项:

  • 使Car不实现Vehicle(任何接口)。这样,bean将由CGLIB代理(并且其类保持不变),并且您的代码将正常工作
  • 通过在注释@EnableAspectJAutoProxy(proxyTargetClass=true)中添加以下属性来强制在各处使用CGLIB代理。您的代码可以正常工作
  • 按名称获取bean(请参见上面的代码)
  • 通过接口获取bean:
    Vehicle car =  context.getBean(Vehicle.class);      
    car.drive();