Hibernate【與Spring整合】
前言
前面已經學習瞭如何使用Spring與Struts2進行整合,本博文主要講解如何使用Spring對Hibernate進行整合
Spring和Hibernate整合的關鍵點:
- SessionFactory物件交給Spring來建立
- Hibernate的事務交給Spring進行管理
Spring和Hibernate整合步驟
引入jar包
- 連線池/資料庫驅動包
- Hibernate相關jar
- Spring 核心包(5個)
- Spring aop 包(4個)
- spring-orm-3.2.5.RELEASE.jar 【spring對hibernate的支援】
- spring-tx-3.2.5.RELEASE.jar 【事務相關】
這裡寫圖片描述
配置檔案
- hibernate.cfg.xml
- bean.xml
bean.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> </beans>
hibernate.cfg.xml
<hibernate-configuration> <!-- 通常,一個session-factory節點代表一個數據庫 --> <session-factory> <!-- 1. 資料庫連線配置 --> <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property> <property name="hibernate.connection.url">jdbc:mysql:///zhongfucheng</property> <property name="hibernate.connection.username">root</property> <property name="hibernate.connection.password">root</property> <!-- 資料庫方法配置, hibernate在執行的時候,會根據不同的方言生成符合當前資料庫語法的sql --> <property name="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</property> <!-- 2. 其他相關配置 --> <!-- 2.1 顯示hibernate在執行時候執行的sql語句 --> <property name="hibernate.show_sql">true</property> <!-- 2.2 格式化sql --> <property name="hibernate.format_sql">true</property> <!-- 2.3 自動建表 --> <property name="hibernate.hbm2ddl.auto">create</property> </session-factory> </hibernate-configuration>
搭建配置環境測試
- User
package bb;
/**
* Created by ozc on 2017/5/15.
*/
public class User {
private String name;
private String password;
private int id;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
@Override
public String toString() {
return "User{" +
"name='" + name + ''' +
", password='" + password + ''' +
'}';
}
}
- IUser介面
public interface IUser {
void save();
}
- UserDao
public class UserDao implements IUser {
@Override
public void save() {
}
}
- userService
public class UserService {
private UserDao userDao;
public void save() {
userDao.save();
}
}
測試Spring環境
首先,我們為userDao、userService使用Spring來建立物件,以及新增物件的依賴關係,看看Spring的環境是否成功
- 建立UserDao例項--->@Repository
@Repository
public class UserDao implements IUser {
@Override
public void save() {
}
}
- 建立userService例項,並注入userDao屬性
@Service
public class UserService {
@Autowired
private UserDao userDao;
public void save() {
userDao.save();
}
}
- 在Spring配置檔案中使用註解掃描器
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="bb"/>
</beans>
- 測試:成功得到userService物件,並且userService物件含有userDao屬性的值
public class Test2 {
@Test
public void test33() {
ApplicationContext ac = new ClassPathXmlApplicationContext("spring-config.xml");
UserService userService = (UserService) ac.getBean("userService");
System.out.println(userService);
}
}
這裡寫圖片描述
測試Hibernate環境
- 對映配置檔案
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="bb">
<class name="User" table="t_user">
<id name="id" column="user_id">
<generator class="native"></generator>
</id>
<property name="name" column="name"></property>
<property name="password" column="password"></property>
</class>
</hibernate-mapping>
- 主配置檔案載入對映檔案
<mapping resource="bb/User.hbm.xml" />
- 建立SessionFactory,Session
@Repository
public class UserDao implements IUser {
@Override
public void save(User user) {
//得到SessionFactory
SessionFactory sessionFactory = new Configuration().configure().buildSessionFactory();
//得到Session
Session session = sessionFactory.openSession();
session.beginTransaction();
session.save(user);
session.getTransaction().commit();
session.close();
}
}
- 測試:
public class Test2 {
@Test
public void test33() {
ApplicationContext ac = new ClassPathXmlApplicationContext("spring-config.xml");
UserService userService = (UserService) ac.getBean("userService");
userService.save(new User());
}
}
這裡寫圖片描述
使用Spring建立SessionFactory物件
Spring與Hibernate整合的關鍵點之一就是使用Spring來建立SessionFactory物件。其中又有三種方式來建立SessionFactory
直接載入hibernate主配置檔案
<!--
SessionFactory是一個工廠,我們要使用它的實現類
我們使用的是hibernate的3.6版本,因此載入的是3
-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<!--說明配置檔案所在的位置-->
<property name="configLocation" value="classpath:hibernate.cfg.xml"/>
</bean>
那麼在userDao中就不用我們自己手動來建立SessionFactory物件了。
@Repository
public class UserDao implements IUser {
@Autowired
private SessionFactory sessionFactory;
@Override
public void save(User user) {
//得到Session
Session session = sessionFactory.openSession();
session.beginTransaction();
session.save(user);
session.getTransaction().commit();
session.close();
}
}
這裡寫圖片描述
連線池交給Spring管理
我們知道Hibernate對C3P0的連線池支援度比不上Spring,因此我們可以使用Spring的連線池。因此我們載入Hibernate的主配置檔案又使用Spring的資料庫連線池
也就是說,一部分配置在hibernate.cfg.xml,一部分配置在Spring檔案中
<!-- 資料來源配置 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
<property name="jdbcUrl" value="jdbc:mysql:///zhongfucheng"></property>
<property name="user" value="root"></property>
<property name="password" value="root"></property>
<property name="initialPoolSize" value="3"></property>
<property name="maxPoolSize" value="10"></property>
<property name="maxStatements" value="100"></property>
<property name="acquireIncrement" value="2"></property>
</bean>
<!--
載入Hibernate的主配置檔案,又使用Spring的資料庫連線池
-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<!--說明配置檔案所在的位置-->
<property name="configLocation" value="classpath:hibernate.cfg.xml"/>
<property name="dataSource" ref="dataSource"/>
</bean>
這裡寫圖片描述
配置檔案全寫Spring中【推薦】
上面我們一部分是載入Hibernate的主配置檔案,一部分是使用Spring配置檔案的資料庫連線池…這樣不好…我們應該在Spring中對其進行同一的管理!
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 資料來源配置 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
<property name="jdbcUrl" value="jdbc:mysql:///zhongfucheng"></property>
<property name="user" value="root"></property>
<property name="password" value="root"></property>
<property name="initialPoolSize" value="3"></property>
<property name="maxPoolSize" value="10"></property>
<property name="maxStatements" value="100"></property>
<property name="acquireIncrement" value="2"></property>
</bean>
<!--
所有的配置資訊都在Spring中完成。
-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<!--Hibernate常用的配置屬性-->
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
<prop key="hibernate.show_sql">true</prop>
<prop key="hibernate.hbm2ddl.auto">update</prop>
</props>
</property>
<!--Hibernate載入對映檔案,對映到資料夾-->
<!-- <property name="mappingDirectoryLocations">
<list>
<value>bb</value>
</list>
</property>-->
<!--Hibernate載入對映檔案,對映到具體位置-->
<property name="mappingLocations">
<list>
<value>bb/User.hbm.xml</value>
</list>
</property>
</bean>
<context:component-scan base-package="bb"/>
</beans>
我們推薦的就是使用這一種,就可以少了Hibernate的配置檔案了。並且容易統一管理。
Spring管理事務
到目前為止,我們是使用Hibernate程式設計式事務控制管理,Spring與Hibernate整合另一個關鍵就是使用Spring對Hibernate進行事務管理
<!--配置Hibernate的事務管理器類-->
<bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<!--引用的是SessionFactory,SessionFactory包括了資料連線池-->
<property name="sessionFactory" ref="sessionFactory"/>
</bean>
<!--開啟以註解的方式來管理事務-->
<tx:annotation-driven transaction-manager="txManager"/>
值得注意的是:Spring與Hibernate整合,Spring只支援執行緒的Session,並且不用我們手動配置
這裡寫圖片描述
userDao
@Repository
public class UserDao implements IUser {
@Autowired
private SessionFactory sessionFactory;
@Override
public void save(User user) {
sessionFactory.getCurrentSession().save(user);
}
}
userService新增@Transactional註解就是為Hibernate添加了事務管理了。
@Service
@Transactional
public class UserService {
@Autowired
private UserDao userDao;
public void save(User user) {
userDao.save(user);
}
}