spring与hibernate整合之:继承HibernateDAOSupport方式+理解xml与AutoWired、Resource的区别

时间:2022-07-29 19:27:28

将spring与hibernate进行整合之后,我们都希望用spring来管理DAO层,这样有利于快速实现功能,少出错。

今天在研究在DAO层编码,采用继承HibernateDAOSupport方式时,发现网上许多人说的都是错的,既然已经继承DAOSupport了,就不应该在DAO层中再有对SessionFactory的注入了。如果注入了,不跟没有用DAOSupport一样了么?或者接近于hibernateTemplate方法了。


我想,既然用了DAOSupport,它里面有了getHibernateTemplate和setSessionFactory方法,且是final的,那么我们就应该直接将SessionFacotry注入到里面。

后来,几经查阅,原来问题在这里。

在spring的配置文件中:写了sessionFactory的bean之后,继承HibernateDAOSupport,发现会报错,说"sessionFactory " or "hibernateTemplate " is required。。。


原因在于,spring没有能将我们在配置文件中写的bean注入到HibernateDAOSupport中,因为spring默认情况下是不会自动装配的。

如果我们指定了默认装配,通过看spring原码,发现它调用了HibernateDAOSupport中的setSessionFactory方法,然后生成了一个hibernateTemplate。进而我们可以使用。


如果再研究一下,会发现一个比较怪的现象:

在spring中的bean中,它的class是:<bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">

但看spring的源码,它的HibernateDAOSupport类中的setSessionFactory方法中,SessionFactory的类型是org.hibernate.SessionFactory。

这到底是如何转换的呢?又看spring的源码,发现了问题。


原来,AnnotationSessionFactoryBean继承了LocalSessionFactoryBean,而在LocalSessionFactoryBean中,有方法buildSessionFactory,它返回值的类型就是org.hibernate.SessionFactory.通过走断点,我们也的确看到它是走的这个方法。


因此呢,spring是通过自己的方法,实现了hibernate的sessionFacotry。也正是如此,我们既可以用byName,也可以用byType,让它能装配就可以了。


解决方法一:最简单的就是在spring的配置文件的头部,写上:default-autowire="byName">或者byType  如果是byName,SessionFactory的bean名字必须是sessionFactory.

 

解决方法二:在Dao层加上:

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

虽然final方法不能重写,但是可以调用,而在调用的时候,把sessionFactory给注入进去就可以了。


也正是解决方法二,让我进一步了解到了spring的xml与annotation的区别:

二者都是为我们进行注入操作,但二者是有重要区别的。

在xml,spring是通过bean的id(XXX),然后调用被注入类的setXXX的方法,如此进行注入。

而在annotation的autowired中,它是通过方法中参数的类来进行注入。也就是setYYY(XXX x) 中的XXX与bean的配置进行对应。

这也就理解了方法二可以正确运行的原因。

注:在required中,如果没有指定名字,则是根据把set后面的第一个字母小写后,当作名字来寻找bean.找不到,则按type找。

 

解决方法三:由于我们不能对其注入的主要原因是,hibernateDaoSupport中的set方法是final的,我们不能重写。因此,我们可以尝试对于这个bean,我们不写@component标签,而是通过xml方式进行注入。

因此可以在配置文件中写上:

<bean name="userDAO" class="com.msb.dao.UserDAOImpl">
    <property name="sessionFactory" ref="mySessionFactory"></property>
   </bean>

在实现层直接就写:

public class UserDAOImpl extends HibernateDaoSupport implements UserIDAO{
     public void save(User user){
         this.getHibernateTemplate().save(user);
    }
}

就可以了。

 

解决方法四:用Annotation中的Resouce标签:(先按名字寻找,再按类型查找)

@Component
 /*
  * 由于HibernateDaoSupport中的setSessionFactory与setHibernateTemplate都是final的。不能重写。
 */
public class UserDAOImpl extends HibernateDaoSupport implements UserIDAO{
 
 @Resource(name="hibernateTemplate")
 public void setSuperHibernateTemplate(HibernateTemplate hibernateTemplate){
  super.setHibernateTemplate(hibernateTemplate);
 }
 
 public void save(User user){
  this.getHibernateTemplate().save(user);
 }
}

 <bean id="hibernateTemplate" >
   <property name="sessionFactory" ref="mySessionFactory"></property>
  </bean>