【问题标题】:Add property in component scan在组件扫描中添加属性
【发布时间】:2018-02-16 16:06:48
【问题描述】:

我们在项目中有很多 DAO 和他们自己的实现。例如,我有一个名为 CorDAO 和 CorDAOHibernate 的类(CorDAO 它是一个接口,而 CorDAOHibernate 实现了 CorDAO)。对于每个接口我需要这样做:

    <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
            <property name="dataSource" ref="dataSource"/>
            <property name="packagesToScan" value="com.myproject.*.to"/>
            <property name="hibernateProperties">
                <props>
                    <prop key="hibernate.dialect">${hibernate.dialect}</prop>
                    <prop key="hibernate.hbm2ddl.auto">update</prop>
                    <prop key="hibernate.show_sql">false</prop>
                    <prop key="hibernate.cache.use_query_cache">true</prop>
                    <prop key="hibernate.cache.use_second_level_cache">true</prop>
                    <prop key="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</prop>
                </props>
            </property>
        </bean>

<bean id="genericDAO" class="com.myproject.dao.hibernate.GenericDAOHibernate">
        <property name="sessionFactory" ref="sessionFactory"/>
    </bean>

        <bean id="CorDAO" class="com.myproject.dao.hibernate.CorDAOHibernate">
                <property name="sessionFactory" ref="sessionFactory"/>
        </bean>
        <bean id="ExampleDAO" class="com.myproject.dao.hibernate.ExampleDAOHibernate">
                <property name="sessionFactory" ref="sessionFactory"/>
        </bean>

这就是为什么我想在 com.myproject.dao.hibernate.* 包中使用 ComponentScan。请注意,CorDAO 它是一个 @Repository 并且 CorDAOHibernate 没有任何注释。 每个 DAO 类都是 GenericDAO 的子类。

如何使用 ComponentScan?我正在使用 Spring 3,所以我尝试了这个:

 <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="packagesToScan" value="com.myproject.*.to"/>
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">${hibernate.dialect}</prop>
                <prop key="hibernate.hbm2ddl.auto">update</prop>
                <prop key="hibernate.show_sql">false</prop>
                <prop key="hibernate.cache.use_query_cache">true</prop>
                <prop key="hibernate.cache.use_second_level_cache">true</prop>
                <prop key="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</prop>
            </props>
        </property>
    </bean>

    <context:component-scan base-package="com.myproject.dao.hibernate" />

我在 CorDAOHibernate 和 GenericDAOHibernate 中添加了一个 @Component 注释,但我得到了一个 IllegalArgumentException:

org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'corDAOHibernate' defined in URL [jar:file:/home/danielamorais/Documents/apache-tomcat-8.0.47/webapps/myproject/WEB-INF/lib/myproject-3.1.0.jar!/com/myproject/dao/hibernate/corDAOHibernate.class]: Invocation of init method failed; nested exception is java.lang.IllegalArgumentException: 'sessionFactory' or 'hibernateTemplate' is required

我认为这是因为 sessionFactory 在 GenericDAOHibernate 中自动装配,但在 CorDAOHibernate 中为空。如何通过组件扫描解决此问题?

类 CorDAO

@Repository
public interface CorDAO extends GenericDAO {
    //Methods
}

CorDAOHibernate

@Component
public class CorDAOHibernate extends GenericDAOHibernate implements CorDAO {
    //Methods
}

通用DAO

public interface GenericDAO extends DAO {
    //Methods. This class doesn't contains @Repository annotation
}

通用DAOHibernate

    @Component
    public class GenericDAOHibernate extends HibernateDaoSupport implements GenericDAO, PropertySelector {

        @Autowired
        private SessionFactory sessionFactory;

    protected Session getCurrentSession() {
        return sessionFactory.getCurrentSession();
    }
        //Methods
    }

public interface DAO {
//There's no method here.
}

另外,添加 @Component("genericDAO") 我得到了(可能是因为继承问题):

NoUniqueBeanDefinitionException: No qualifying bean of type [com.myproject.dao.GenericDAO] is defined: expected single matching bean but found 88

【问题讨论】:

  • 你能发布一个 CorDAOHibernate 的例子吗?
  • 谢谢,所以你需要去掉 GenericDAOHibernate 类中的 sessionFactory 字段。它扩展了已经有 setSessionFactory for you 的 HibernateDaoSupport
  • 子类是否可以访问 private SessionFactory sessionFactory 字段?如果是这样,请添加它是如何实现的。如果像GenericDAOHibernate 这样的具体类无法访问此字段,则可能是个问题
  • CorDAOHibernate的构造函数是什么? (你知道,@Bean/@Component/etc 需要零参数或自动装配的构造函数!?)
  • 而且第二条错误信息很奇怪:You defined 88 GenericDao"beans"!?

标签: java spring hibernate


【解决方案1】:

嗯,你的主要问题是因为这个:

init 方法调用失败;嵌套异常是 java.lang.IllegalArgumentException: 'sessionFactory' or 'hibernateTemplate' is required

我也回答了类似的问题before

长话短说,您的GenericDAOHibernate 以错误的方式实现:

  1. 无需在内部使用SessionFactory,因为HibernateDaoSupport 已经为您实现,您可以通过调用getSessionFactory() 获得
  2. 您必须通过构造函数将SessionFactory 传递给HibernateDaoSupport,以便在HibernateDaoSupport 中初始化HibernateTemplate。否则,它将为 null,因此会发生此异常消息。

另外,您的@Repository@Component 似乎也被注释为尴尬。基本上,我只会将 @Repository 用于 DAO 。它与 @Component 加上 spring "exception translation" 功能相同。

无论如何,请尝试更改为以下内容:

@Repository  
public class GenericDAOHibernate extends HibernateDaoSupport implements GenericDAO, PropertySelector {

         @Autowired
         public GenericDAOHibernate(SessionFactory sessionFactory){
             super.setSessionFactory(sessionFactory);
         }

        protected Session getCurrentSession() {
            return getSessionFactory().getCurrentSession();
        }

}     

以及每个 DAO 实现:

@Repository
public class CorDAOHibernate extends GenericDAOHibernate implements CorDAO {
     public CorDAOHibernate(SessionFactory sessionFactory) {
          super(sessionFactory);
     }
}   

另外,从CorDAO 接口中删除@Repository@Component("genericDAO") 的东西.....

【讨论】:

    【解决方案2】:

    您必须将声明为sessionFactory 的类注释为@Bean(或@Component,请参阅this question,并确保将通过ComponentScan 对其进行扫描。

    还可以考虑使用 Spring Data 中的 Repository 方法,这样您就不必编写样板代码。

    【讨论】:

    • 她已经提到 @Component 是在类上分配的,你可以看到组件扫描也打开了。
    • @SergeyProkofiev 好吧,我没有看到更新。对不起
    【解决方案3】:

    我认为你应该删除

    @自动连线 私有 SessionFactory sessionFactory;

    字段。

    因为 sessionFactory 字段已经在父类中定义。并且 setSessionFactory 方法被取消为 final。所以你不需要为 sessionFactory 清除新的私有字段。

    这里是setSessionFactory方法的实现。

    /**
     * Set the Hibernate SessionFactory to be used by this DAO.
     * Will automatically create a HibernateTemplate for the given SessionFactory.
     * @see #createHibernateTemplate
     * @see #setHibernateTemplate
     */
    public final void setSessionFactory(SessionFactory sessionFactory) {
        if (this.hibernateTemplate == null || sessionFactory != this.hibernateTemplate.getSessionFactory()) {
            this.hibernateTemplate = createHibernateTemplate(sessionFactory);
        }
    }
    

    同时删除 sessionFactory 和 getCurrentSession(HibernateDaoSupport 已经有 currentSession() 方法)

    或者如下改变getCurrentSession

    public class GenericDAOHibernate extends HibernateDaoSupport implements GenericDAO, PropertySelector {
    
    
    
    protected Session getCurrentSession()  throws DataAccessResourceFailureException  {
        return currentSession()
    }
        //Methods
    }
    

    【讨论】:

      猜你喜欢
      • 2014-10-11
      • 1970-01-01
      • 2023-01-19
      • 2018-09-11
      • 2016-10-21
      • 1970-01-01
      • 2019-07-19
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多