在Web项目中一般会把各个web框架结合在一起使用,比如spring+hibernate,spring+ibatis等,如此以来将其他的框架整合到spring中来,便有些少许的不便,当然spring已经把这种整合变得很简单了。本人结合在项目中使用过的spring和ibatis,以及和mybatis的整合,进而小测了一下和hibernate的整合,望批评指正。
一、spring、hibernate整合
Spring中不但可以选择SpringJDBC作为持久化技术,还可以选择Hibernate、iBatis、JPA、JDO等多种类型的持久化技术。Spring提供了方便的模板类对原ORM进行简化封装。让我们先回忆一下单独使用Hibernate时的过程,我们需要编写好一个对象关系的映射文件命名形式如xxx.hbm.xml,然后通过Hibernate的配置文件hibernate.cfg.xml 将所有的xxx.hbm.xml 映射文件组装起来,最后通过两行经典的代码得到SessionFactory的实例,如
Configuration cfg = new Configuration().configure("hiberante.cfg.xml");
SessionFactory sessionFactory = cfg.buildSessionFactory();
如果使用注解的话使用AnnotationConfiguration对象,现在Spring为创建SessionFactory提供了一个好用的FactoryBean工厂类:org.springframework.orm.hibernate3.LocalSessionFactoryBean,通过配置一些必要的属性,即可获取一个SessionFactoryBean。
1.使用HibernateTemplate
Hibernate.cfg.xml文件中配置了数据源、对象关联映射文件以及Hibernate控制属性信息。因此集成到spring中以后要把该文件中内容都拿过来如下:
<context:componet-scan base-package="com.demo" />
<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.org.hiberante3.LocalSessionFactoryBean" >
<property name="dataSource" ref="dataSource"/>
<property name="mappingResources" >
<list>
<value>com.demo.hibernate.domain/xxx.hbm.xml</value>
</list>
</property>
<property name="hiberanteProperties">
<props>
<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
<prop key="hibernate.show_sql">true</prop>
</props>
</property>
</bean>
<bean id="hiberanteTemplate" class="org.springframework.orm.hiberante3.HibernateTemplate">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>
<bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>
<tx:annotation-driven transaction-manager="transactionManager"/>
如上面的配置所示,基于模板类使用hibernate比较简单,spring提供了使用模板的支持类HibernateDaoSupport类,并通过getHibernateTemplate()方法向子类开放模板类实例的调用。如果不想在每个dao里面都加上HibernateTemplate,可以将其放在一个BaseDao里面,然后通过扩展该类创建一个使用HibernateTemplate的Dao,使用时直接getHibernateTemplate().op(obj) 就可以了。也可以将我们自己的Dao实现直接继承HibernateDaoSupport类同时实现Dao接口,该方式也很简单。剩下的就和单独使用Hibernate区别很小了。
对于lob类型的数据,我们还需要在Spring配置文件重定一个Lob数据处理器,让SessionFactory拥有处理Lob数据的能力。在SessionFactory里引用该bean:
<bean id="lobHandler" class="org.springframework.jdbc.support.lob.DefaultLobHandler" lazy-init="true" />
我们可以通过注入SessionFactory 直接通过getCurrentSession()能够获取和当前线程绑定的Session。Hibernate自身具备了获取和事务线程绑定的Session对象的功能,这与在Spring的HibernateTemplate中使用和事务绑定的Session相同。因此使用原生的API,同样和spring和事务管理器一起工作。和使用template不同的是原生API抛出的异常是Hibernate异常。
3.使用注解
Hibernate通过AnnotationConfiguration的addAnnotatedClass和addPackage方法加载使用JPA注解的实体类,获取映射的元数据信息,并在此基础上创建SessionFactory实例。spring专门提供了一个配套的AnnotationSessionFactoryBean,用以创建基于JPA注解的SessionFactory,该类扩展了LocalSessionFactoryBean类,增强的功能是可以根据实体类的注解获取ORM的配置信息。也允许混合使用XML配置和注解配置对象关系映射,Hibernate内部自动整合这些元数据信息,并不会产生冲突。
此外,Spring以防Web层访问Service层延迟加载的对象,专门提供了一个OpenSessionInViewFilter过滤器,它的主要功能是使每个请求过程绑定一个Hibernate Session,即使最初的事务已经完成了,也可以在Web层进行延迟加载的操作。OpenSessionInViewFilter过滤器将HibernateSession绑定到请求线程中,它将自动被Spring的事务管理器探测到,因此其使用与使用HibernateTransactionManager或JtaTransactionManager进行事务管理的环境,也可以用于非只读事务的数据操作中。
二、spring集成ibatis/mybatis
ibatis的后续版本改名叫mybatis,现在两者都被大家经常使用。两者的最大不同个人感觉是mybatis提供了接口绑定,你不需要再去实现Dao接口,而直接可在Service层调用,将注意力集中在映射文件上即可。ibatis的核心类是SqlMapClient,XmlSqlMapClientFactoryBean通过读取文件中的sqlMapConfig(当然包括很多的sqlMap),构建一个SqlMapClient,其通过sqlMap来执行映射的每个SQL语句;mybatis的核心类是SqlSessionFactory,SqlSessionFactoryBuilder读取配置文件(或注解)创建一个SqlSessionFactory,其通过SqlSession来执行已经映射的SQL语句。下面是两者和spring集成的配置:
1.Ibatis需要自己在单独注入各个DAO文件,如
<bean id= "xxxDao" class ="com.demo.ibatis.dao.impl.xxxDaoImpl">
<property name ="sqlMapClient" ref="sqlMapClient" />
</bean >
<bean id ="yyyDao" class= "com.demo.ibatis.dao.impl.yyyDaoImpl" >
<property name ="sqlMapClient" ref="sqlMapClient" />
</bean >
<bean id ="zzzDao" class= "com.demo.ibatis.dao.impl.zzzDaoImpl" >
<property name ="sqlMapClient" ref="sqlMapClient" />
</bean >
在spring-prop.xml文件中配置sqlMapClient,如
<bean id= "sqlMapClient" class= "org.springframework.orm.ibatis.SqlMapClientFactoryBean" >
<property name ="configLocation" value= "classpath:ibatis/ibatis-config.xml" />
<property name ="dataSource" ref="clopsDataSource" />
<property name ="lobHandler" ref="lobHandler"/>
</bean >
如property指定的configuration的位置一样,要有<sqlMapConfig>的配置,并将各个xml文件引入进来,ibatis-config.xml如下:
<sqlMapConfig>
<settings useStatementNamespaces= "true" cacheModelsEnabled= "true" enhancementEnabled= "true"
lazyLoadingEnabled= "true" maxRequests= "32" maxSessions= "10" maxTransactions= "5" />
<sqlMap resource ="ibatis/xxx.xml" />
<sqlMap resource ="ibatis/yyy.xml" />
<sqlMap resource ="ibatis/zzz.xml" />
......
</sqlMapConfig>
2.而在Mybatis中则不需要这样配置:
在spring-prop中指定sqlSessionFactory,如,
<bean id= "sqlSessionFactory" class= "org.mybatis.spring.SqlSessionFactoryBean" >
<property name ="dataSource" ref="dataSource" />
<property name ="typeAliasesPackage" value= "com.demo.mybatis.model" />
<property name ="mapperLocations" value= "classpath*:mapper/**/*.xml" />
</bean >
<bean id ="sqlSession" class= "org.mybatis.spring.SqlSessionTemplate" >
<constructor-arg index ="0" ref="sqlSessionFactory" />
<constructor-arg index ="1" value="BATCH" />
</bean >
<bean class ="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name ="basePackage" value= "com.demo.mybatis.dao" />
<property name ="sqlSessionTemplate" ref= "sqlSession" />
</bean >
其中各种mapper文件在由mapperLocations指定。mybatis相对简单些。
使用时类似上面的Hibernate,ibatis可将sqlMapClientTemplate直接注入,或者继承SqlMapClientDaoSupport类同时实现Dao接口,通过getSqlMapClientTemplate()方法或者sqlMapClient进行操作。注意上述ibatis直接注入的是sqlMapClient,其实sqlMapClientTemplate的构造函数引入了一个sqlMapClient,其使用的方法均来自sqlMapClient。
Mybatis与spring的集成因对数据处理的不同方式而不同,上述方式是采用MapperScannerConfigurer的处理方式,且采用了注解形式,还可以使用MapperFactoryBean、SqlSession、SqlSessionDaoSupport的数据处理方式,使用后两种方式时需要自己实现Dao接口,而对于MapperFactoryBean其直接指定Dao接口mapperInterface。