最近一直在研究S2SH開發框架,把三部分分別看完之后開始整合,發現由於 hibernate 高版本已經又完善了好多功能,導致與之前跟 spring 整合的方法有些出入。
先說一下與 hibernate3 整合的時候獲取 Session 的辦法吧。
以檢測用戶名和密碼是否正確為例,數據表是users,所以使用MyEclipse反向生成其所對應的類和.hbm.xml文件分別是Users.java和Users.hbm.xml。
我是直接使用 MyEclipse 添加的兩個框架,所以系統自動給生成了 applicationContext.xml 配置文件,關於 SessionFactory 的部分如下:
1 <bean id="sessionFactory" 2 class="org.springframework.orm.hibernate4.LocalSessionFactoryBean"> 3 <property name="dataSource"> 4 <ref bean="dataSource" /> 5 </property> 6 <property name="hibernateProperties"> 7 <props> 8 <prop key="hibernate.dialect"> 9 org.hibernate.dialect.MySQLDialect 10 </prop> 11 </props> 12 </property> 13 <property name="mappingResources"> 14 <list> 15 <value>com/ssh/entity/Users.hbm.xml</value> 16 </list> 17 </property> 18 </bean>
其實在這里也可以看到兩個框架的整合,在sessionFactory中添加了 mappingResource 的<property>標簽,實現OR映射。
如果使用hibernate3提供的接口,那么在DAO中的類要繼承HibernateDaoSupport類,那么就可以使用:
1 super.getHibernateTemplate().executeFind(new HibernateCallback() { 2 public Object doInHibernate(Session session)throws HibernateException, SQLException { 3 ...... 4 } 5 }
可以看到在 doInHibernate()中的參數就是session,那我們就可以直接使用了。
關鍵就是現在它不是出hibernate4了么,很多人就是有強迫症,總喜歡用最高級版的,那就用唄,然后就還按照之前的這個方法去做了,結果拋異常了。。。我就是強迫症患者之一。。。異常如下:
1 Exception in thread "main" java.lang.NoSuchMethodError: org.hibernate.SessionFactory.openSession()Lorg/hibernate/classic/Session; 2 at org.springframework.orm.hibernate3.SessionFactoryUtils.doGetSession(SessionFactoryUtils.java:324) 3 at org.springframework.orm.hibernate3.SessionFactoryUtils.getSession(SessionFactoryUtils.java:235) 4 at org.springframework.orm.hibernate3.HibernateTemplate.getSession(HibernateTemplate.java:457) 5 at org.springframework.orm.hibernate3.HibernateTemplate.doExecute(HibernateTemplate.java:393) 6 at org.springframework.orm.hibernate3.HibernateTemplate.executeFind(HibernateTemplate.java:343)
很奇怪的一件事就是明明使用的hibernate4,怎么異常信息都是關於hibernate3的?然后又返回去看了一下繼承的HibernateDaoSupport,其實是:
import org.springframework.orm.hibernate3.support.HibernateDaoSupport;
后來上網查了一下,主要原因就是hibernate4已經提供了很好的機制,所以不需要再繼承HibernateDaoSupport類了。那怎么辦呢?不就是為了獲取到Session么,我們直接在UserDAOImpl.java類中添加一個SessionFactory屬性,再添上它的setter方法。我們使用的DAO層的實現層是要添加到applicationContext.xml中的,其實也就是spring的依賴注入,比如我的UserDAOImol.java在xml中的添加如下:
1 <bean id="userDAO" class="com.ssh.dao.UserDAOImpl"> 2 <property name="sessionFactory" ref="sessionFactory"></property> 3 </bean>
這里的sessionFactory在上面已經提到了,這是系統自動生成的。看到這里的代碼就應該會想到,其實到這里我們已經獲取到了sessionFactory實例。至於它的使用肯定已經很熟悉了,不多說。我的UserDAOImpl.java代碼如下:
1 package com.ssh.dao; 2 3 import java.util.List; 4 5 import org.hibernate.Criteria; 6 import org.hibernate.Session; 7 import org.hibernate.SessionFactory; 8 import org.hibernate.criterion.Restrictions; 9 10 import com.ssh.entity.Users; 11 12 public class UserDAOImpl implements UserDAO { 13 private SessionFactory sessionFactory; 14 15 public void setSessionFactory(SessionFactory sessionFactory) { 16 this.sessionFactory = sessionFactory; 17 } 18 19 public List search(final Users condition) { 20 //Session session = sessionFactory.getCurrentSession(); 21 Session session = sessionFactory.openSession(); 22 Criteria criteria = session.createCriteria(Users.class); 23 if (condition != null) { 24 String username = condition.getUsername(); 25 String password = condition.getPassword(); 26 if (username != null && !username.equals("")) { 27 criteria.add(Restrictions.eq("username", username)); 28 } 29 if (password != null && !password.equals("")) { 30 criteria.add(Restrictions.eq("password", password)); 31 } 32 } 33 return criteria.list(); 34 35 } 36 37 }
一定要注意這里第20行的注釋部分,不要用getCurrentSession(),不然還是有異常。直接用openSession()