Синглтон и @Autowired возвращают NULL - PullRequest
10 голосов
/ 16 февраля 2011

У меня есть менеджер репозитория, который управляет моими репозиториями. У меня есть @Autowired для создания экземпляров моих свойств, но они всегда нулевые. Бины правильно настроены в моем XML. Любая причина почему?

public class RepositoryManager {

        private static RepositoryManager instance;

        private RepositoryManager()
        {
        }

       public static RepositoryManager Instance()
       {
            if(instance == null)
                 instance  = new RepositoryManager();

            return instance;
        }

        @Autowired
        private IUserRepository userRepository;

        @Autowired
        private IRoleRepository roleRepository;

        @Autowired
        private IAssetRepository assetRepository;

        public IUserRepository getUserRepository() {
            return userRepository;
        }

        public void setUserRepository(IUserRepository userRepository) {
            this.userRepository = userRepository;
        }

        public IRoleRepository getRoleReposttory() {
            return roleRepository;
        }

        public void setRoleReposttory(IRoleRepository roleRepository) {
            this.roleRepository = roleRepository;
        }

        public IAssetRepository getAssetRepository() {
            return assetRepository;
        }

        public void setAssetRepository(IAssetRepository assetRepository) {
            this.assetRepository = assetRepository;
        }
    }

dao.xml

    <!-- Scans within the base package of the application for @Components to configure as beans -->
    <context:component-scan base-package="com.cisco.badges.data.*" />

    <context:property-placeholder location="classpath:jdbc.properties"/>

    <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
        <property name="driverClassName" value="${jdbc.driverClassName}"/>
        <property name="url" value="${jdbc.url}"/>
        <property name="username" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>
      </bean>

    <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="configurationClass" value="org.hibernate.cfg.AnnotationConfiguration"/>
        <property name="annotatedClasses">
          <list>
              <value>com.cisco.badges.data.domain.User</value>
              <value>com.cisco.badges.data.domain.Role</value>
              <value>com.cisco.badges.data.domain.Asset</value>
              <value>com.cisco.badges.data.domain.UserRole</value>
              <value>com.cisco.badges.data.domain.UserRole$UserRolePK</value>
              <value>com.cisco.badges.data.domain.UserAsset</value>
              <value>com.cisco.badges.data.domain.UserAsset$UserAssetPK</value>
          </list>
        </property>
        <property name="configLocation">
            <value>classpath:hibernate.cfg.xml</value>
        </property>
        <property name="hibernateProperties">
            <props>
                 <prop key="hibernate.show_sql">true</prop>
                <prop key="hibernate.format_sql">true</prop>
            </props>
        </property>
    </bean>

     <tx:annotation-driven/>

     <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory" />
    </bean>

</beans>

userrepository.java

    @Repository("userRepository")
public class UserRepository extends
        BaseRepository<User, Long> implements
        IUserRepository {

    @Autowired
    public UserRepository(SessionFactory sessionFactory) {
        super(sessionFactory);
    }

Ответы [ 4 ]

22 голосов
/ 30 сентября 2011

Я просто столкнулся с этим сам. Проблема в том, что когда вы делаете

new RepositoryManager();

в Instance () вы не используете Spring для создания RepositoryManager, и, следовательно, внедрение зависимостей не происходит для вашего экземпляра (без автопроводки).

Решение состоит в том, чтобы покончить с одноэлементным шаблоном Instance (). Если вы хотите настаивать на синглтоне, сделайте это

@Component
@Scope(value = "singleton")
public class RepositoryManager {
    ...
}

Тогда везде, где вам нужен менеджер репозитория, просто используйте autowire в ссылке на него (при условии, что вызывающим компонентом также управляет Spring!)

@Autowired
private RepositoryManager repositoryManager = null;
5 голосов
/ 16 февраля 2011

Пожалуйста, убедитесь, что у вас есть следующее в вашей конфигурации:

<context:annotation-config />

<context:component-scan base-package="name.of.your.package"></context:component-scan>

Если у вас есть, то опубликуйте свою конфигурацию xml

4 голосов
/ 20 сентября 2013

На самом деле существует очень элегантный способ получить свой торт и съесть его, т. Е. Иметь синглтон JVM, который также управляется Spring.Допустим, у вас есть чистый Java-синглтон с автоматическим компонентом, подобным этому:

public final class MySingletonClass{
  private static MySingletonClass instance;

  public static MySingletonClass getInstance(){
    if(instance==null){
      synchronized{
        if(instance==null){
          instance = new MySingletonClass();
        }
      }
    }
    return instance;
  }

  @Autowired
  private SomeSpringBean bean;

  // other singleton methods omitted
}

Вы можете заставить Spring управлять этим синглтоном, просто добавив в контекст приложения следующую строку:

<bean class="com.mypackage.MySingletonClass" factory-method="getInstance"/>

Теперь в вашем синглтоне будет экземпляр SomeSpringBean с автопроводкой (если он доступен в контексте).

Более того, это «исправление» типичной проблемы с одноэлементными компонентами Spring, которые на самом деле не являются синглетами JVM, посколькуони получают экземпляр от Spring.Использование вышеприведенного шаблона обеспечивает использование одноэлементного уровня JVM, т. Е. Одноэлементного режима, поддерживаемого компилятором, вместе с одноэлементным контейнером.

0 голосов
/ 05 июня 2012

Причина, по которой это происходит, в том, что статический метод Instance()

вы создаете POJO вне контекста пружины.

вы можете это исправить, добавив <context:spring-configured />к вашей конфигурации, а затем аннотировать RepositoryManager с помощью @Configurable

...