Spring @Autowired ОК на Юнит, NPE на основной класс - PullRequest
1 голос
/ 27 марта 2012

У меня есть проект с открытым исходным кодом, размещенный на this github. Я сталкиваюсь со странным сценарием. Я нашел много людей, которые могли заставить аннотацию @Autowired Spring работать в своих основных классах, но не в тестовых классах JUnit. Моя проблема в обратном. Я могу успешно использовать @Autowired в своем тестовом классе JUnit, но когда тест вызывает мой основной класс, в него не вставляются зависимости. Вот мой контекст (упрощенная версия):

Класс входа в систему:

package net.openrally.restaurant.core.exposure.resource;

@Path("/login")
@Component
@Transactional
@Singleton
@Produces("application/json")
@Consumes("application/json")
public class Login extends BaseResource{

    @Autowired
    private UserDAO userDAO;

    @POST
    public Response post(String requestBody){

        ...

        //NullPointerException
        User user = userDAO.loadByCompanyIdAndLogin(companyId, login);
    }

    ...

}

Класс LoginTest:

package net.openrally.restaurant.core.exposure.resource;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("/applicationContext.xml")
public class LoginTest extends BaseResourceTest {

    @Autowired
    private UserDAO userDAO;

    ...

    @Test
    public void testInvalidPassword() {

    ....

    // Works perfectly!
    userDAO.save(user);

    ....

    }

}

applicationContext.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans    http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd">

    <bean id="propertyConfigurer"
        class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
        <property name="locations">
            <list>
                <value>classpath:configuration.properties</value>
            </list>
        </property>
    </bean>

    <context:annotation-config />
    <context:component-scan base-package="net.openrally.restaurant.core" />

    <import resource="db-config.xml" />
</beans>

Я ищу решение уже несколько дней. Из того, что я мог найти до сих пор, большинство людей, у которых есть @Autowired проблемы, не имеют <context:annotation-config /> или xmlns:context="http://www.springframework.org/schema/context" в своем весеннем XML или не имеют @Component семейных аннотаций в классе, который они хотят, чтобы DI имел место и, как вы можете видеть, они оба есть :(. У меня есть только один applicationContext.xml в моем проекте, который подходит как для времени выполнения, так и для теста (у меня есть разные настройки config.properties для установки учетных данных базы данных и уровней журналов по-разному, но пружины нет конфигурация там)

Я использую:

Spring: 3.1.0.RELEASE
JUnit: 4.10
Jersey: 1.11
CLIB: 2.2.2

Любые идеи, и я имею в виду ЛЮБОЙ :), высоко ценится.

UPDATE

При запуске теста появляются следующие журналы:

2012-03-27 07:37:02,457 DEBUG [main] org.springframework.beans.factory.support.DefaultListableBeanFactory - Creating shared instance of singleton bean 'login'
2012-03-27 07:37:02,457 DEBUG [main] org.springframework.beans.factory.support.DefaultListableBeanFactory - Creating instance of bean 'login'
2012-03-27 07:37:02,459 DEBUG [main] org.springframework.beans.factory.annotation.InjectionMetadata - Found injected element on class [net.openrally.restaurant.core.exposure.resource.Login]: AutowiredFieldElement for private net.openrally.restaurant.core.persistence.dao.ConfigurationDAO net.openrally.restaurant.core.exposure.resource.Login.configurationDAO
2012-03-27 07:37:02,459 DEBUG [main] org.springframework.beans.factory.annotation.InjectionMetadata - Found injected element on class [net.openrally.restaurant.core.exposure.resource.Login]: AutowiredFieldElement for private net.openrally.restaurant.core.persistence.dao.UserDAO net.openrally.restaurant.core.exposure.resource.Login.userDAO
2012-03-27 07:37:02,459 DEBUG [main] org.springframework.beans.factory.support.DefaultListableBeanFactory - Eagerly caching bean 'login' to allow for resolving potential circular references
2012-03-27 07:37:02,459 TRACE [main] org.springframework.beans.CachedIntrospectionResults - Getting BeanInfo for class [net.openrally.restaurant.core.exposure.resource.Login]
2012-03-27 07:37:02,462 TRACE [main] org.springframework.beans.CachedIntrospectionResults - Caching PropertyDescriptors for class [net.openrally.restaurant.core.exposure.resource.Login]
2012-03-27 07:37:02,462 TRACE [main] org.springframework.beans.CachedIntrospectionResults - Found bean property 'class' of type [java.lang.Class]
2012-03-27 07:37:02,462 DEBUG [main] org.springframework.beans.factory.annotation.InjectionMetadata - Processing injected method of bean 'login': AutowiredFieldElement for private net.openrally.restaurant.core.persistence.dao.ConfigurationDAO net.openrally.restaurant.core.exposure.resource.Login.configurationDAO
2012-03-27 07:37:02,462 DEBUG [main] org.springframework.beans.factory.support.DefaultListableBeanFactory - Returning cached instance of singleton bean 'configurationDAO'
2012-03-27 07:37:02,462 DEBUG [main] org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor - Autowiring by type from bean name 'login' to bean named 'configurationDAO'
2012-03-27 07:37:02,463 DEBUG [main] org.springframework.beans.factory.annotation.InjectionMetadata - Processing injected method of bean 'login': AutowiredFieldElement for private net.openrally.restaurant.core.persistence.dao.UserDAO net.openrally.restaurant.core.exposure.resource.Login.userDAO
2012-03-27 07:37:02,463 DEBUG [main] org.springframework.beans.factory.support.DefaultListableBeanFactory - Returning cached instance of singleton bean 'userDAO'
2012-03-27 07:37:02,463 DEBUG [main] org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor - Autowiring by type from bean name 'login' to bean named 'userDAO'
2012-03-27 07:37:02,464 DEBUG [main] org.springframework.aop.framework.autoproxy.InfrastructureAdvisorAutoProxyCreator - Creating implicit proxy for bean 'login' with 0 common interceptors and 1 specific interceptors
2012-03-27 07:37:02,464 DEBUG [main] org.springframework.aop.framework.Cglib2AopProxy - Creating CGLIB2 proxy: target source is SingletonTargetSource for target object [net.openrally.restaurant.core.exposure.resource.Login@60532a0a]
2012-03-27 07:37:02,465 DEBUG [main] org.springframework.aop.framework.Cglib2AopProxy - Unable to apply any optimisations to advised method: public javax.ws.rs.core.Response net.openrally.restaurant.core.exposure.resource.Login.post(java.lang.String)

Может ли Spring правильно создавать мой бин, кроме Джерси, с использованием собственного экземпляра без автопроводки?

Ответы [ 2 ]

0 голосов
/ 27 декабря 2013

У меня была точно такая же проблема, и позже я обнаружил, что это проблема разработчика:)

Я создавал новый объект вместо использования bean-компонента.

Допустим, класс MyService автоматически передает класс Дао MyDao myDaoBean. Теперь допустим, что я хочу использовать MyService в MyController, я должен подключиться к бобу Spring myServiceSpringBean. Если я пытаюсь создать новый myServiceObject, то Spring не подключает myDaoBean к myServiceObject, потому что не знает об этом новом объекте службы.

И это приводит к тому, что myDaoBean является нулевым.

0 голосов
/ 15 апреля 2012

Проведя некоторое время, я не смог найти способ заставить GrizzlyWeb использовать контекст, загруженный пружиной, поэтому я попытался найти альтернативное решение.

Затем я нашел Hifaces20 ,что позволит вам запускать и останавливать экземпляр Jetty в рамках одной JVM (это означает, что вы можете, например, использовать базу данных памяти, которая будет видна как вашими тестами, так и вашим приложением)

...