Spring Hibernate集成示例教程(Spring 4 + Hibernate 3和Hibernate 4)

时间:2020-02-23 14:35:52  来源:igfitidea点击:

Spring是最常用的Java EE框架之一,而Hibernate是最流行的ORM框架。
这就是为什么Spring Hibernate组合在企业应用程序中大量使用的原因。
最近,我为Spring教程和Hibernate教程写了很多书,因此关于Spring Hibernate集成的帖子由来已久。

Spring Hibernate

今天,在本教程中,我们将使用Spring 4并将其与Hibernate 3集成,然后更新同一项目以使用Hibernate4。
由于Spring和Hibernate都有很多版本,并且Spring ORM工件同时支持Hibernate 3和Hibernate 4。
,最好列出我在项目中使用过的所有依赖项。

请注意,我注意到所有Spring和休眠版本都不兼容,以下版本对我有用,因此我认为它们兼容。
如果您正在使用其他版本并获取到java.lang.NoClassDefFoundError,则意味着它们不兼容。
主要是因为Hibernate类从一个包移到另一个包导致此错误。
例如,在最新的休眠版本中,将org.hibernate.engine.FilterDefinition类移至org.hibernate.engine.spi.FilterDefinition。

  • Spring Framework版本:4.0.3.3
  • Hibernate Core和Hibernate EntityManager版本:3.6.9.Final和4.3.5.Final
  • Spring ORM版本:4.0.3.3

数据库设置

我正在为项目使用MySQL数据库,因此下面的setup.sql脚本将为该示例创建必要的表。

CREATE TABLE `Person` (
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
`name` varchar(20) NOT NULL DEFAULT '',
`country` varchar(20) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8;
commit;

Spring Hibernate集成示例项目结构

下图显示了最终的项目结构,我们将逐一介绍每个组件。

Maven依赖

我们将首先在pom.xml文件中查找所有必需的依赖项及其版本。

<project xmlns="https://maven.apache.org/POM/4.0.0" xmlns:xsi="https://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="https://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<groupId>org.springframework.samples</groupId>
	<artifactId>SpringHibernateExample</artifactId>
	<version>0.0.1-SNAPSHOT</version>

	<properties>

		<!-- Generic properties -->
		<java.version>1.6</java.version>
		<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
		<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>

		<!-- Spring -->
		<spring-framework.version>4.0.3.RELEASE</spring-framework.version>

		<!-- Hibernate/JPA -->
		<!-- <hibernate.version>4.3.5.Final</hibernate.version> -->
		<hibernate.version>3.6.9.Final</hibernate.version>

		<!-- Logging -->
		<logback.version>1.0.13</logback.version>
		<slf4j.version>1.7.5</slf4j.version>

	</properties>

	<dependencies>
		<!-- Spring and Transactions -->
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-context</artifactId>
			<version>${spring-framework.version}</version>
		</dependency>
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-tx</artifactId>
			<version>${spring-framework.version}</version>
		</dependency>

		<!-- Spring ORM support -->
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-orm</artifactId>
			<version>${spring-framework.version}</version>
		</dependency>

		<!-- Logging with SLF4J & LogBack -->
		<dependency>
			<groupId>org.slf4j</groupId>
			<artifactId>slf4j-api</artifactId>
			<version>${slf4j.version}</version>
			<scope>compile</scope>
		</dependency>
		<dependency>
			<groupId>ch.qos.logback</groupId>
			<artifactId>logback-classic</artifactId>
			<version>${logback.version}</version>
			<scope>runtime</scope>
		</dependency>

		<!-- Hibernate -->
		<dependency>
			<groupId>org.hibernate</groupId>
			<artifactId>hibernate-entitymanager</artifactId>
			<version>${hibernate.version}</version>
		</dependency>
		<dependency>
			<groupId>org.hibernate</groupId>
			<artifactId>hibernate-core</artifactId>
			<version>${hibernate.version}</version>
		</dependency>

		<dependency>
			<groupId>mysql</groupId>
			<artifactId>mysql-connector-java</artifactId>
			<version>5.1.9</version>
		</dependency>
		<dependency>
			<groupId>commons-dbcp</groupId>
			<artifactId>commons-dbcp</artifactId>
			<version>1.4</version>
		</dependency>
	</dependencies>
</project>

Spring和Hibernate集成项目的重要依赖项是:

  • spring-context和spring-tx用于核心Spring功能。
    请注意,我正在使用版本4.0.3.RELEASE。

  • 依赖Spring-orm来获得Spring ORM的支持,在我们的Spring项目中进行休眠集成是必需的。

  • Hibernate框架的hibernate-entitymanager和hibernate-core依赖项。
    注意,版本是3.6.9.Final,要使用Hibernate 4,我们需要将其更改为4.3.5.Final,如上pom.xml文件中所述。

  • mysql-connector-java用于数据库连接的MySQL驱动程序。

模型类或者实体Bean

我们可以使用基于Hibernate XML的映射以及基于JPA注释的映射。
其中我使用JPA注释进行映射,因为hibernate提供了JPA实现。

package com.theitroad.model;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;

/**
 * Entity bean with JPA annotations
 * Hibernate provides JPA implementation
 * @author hyman
 *
 */
@Entity
@Table(name="Person")
public class Person {

	@Id
	@Column(name="id")
	@GeneratedValue(strategy=GenerationType.IDENTITY)
	private int id;
	
	private String name;
	
	private String country;

	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 getCountry() {
		return country;
	}

	public void setCountry(String country) {
		this.country = country;
	}
	
	@Override
	public String toString(){
		return "id="+id+", name="+name+", country="+country;
	}
}

DAO类

我们将在DAO类中实现两个方法,第一个方法将Person对象保存到表中,第二个将从表中获取所有记录并返回Persons列表。

package com.theitroad.dao;

import java.util.List;

import com.theitroad.model.Person;

public interface PersonDAO {

	public void save(Person p);
	
	public List<Person> list();
	
}

上面的DAO类实现如下。

package com.theitroad.dao;

import java.util.List;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;

import com.theitroad.model.Person;

public class PersonDAOImpl implements PersonDAO {

	private SessionFactory sessionFactory;

  public void setSessionFactory(SessionFactory sessionFactory) {
      this.sessionFactory = sessionFactory;
  }
  
	@Override
	public void save(Person p) {
		Session session = this.sessionFactory.openSession();
		Transaction tx = session.beginTransaction();
		session.persist(p);
		tx.commit();
		session.close();
	}

	@SuppressWarnings("unchecked")
	@Override
	public List<Person> list() {
		Session session = this.sessionFactory.openSession();
		List<Person> personList = session.createQuery("from Person").list();
		session.close();
		return personList;
	}

}

注意,这是我们使用Hibernate相关类的唯一地方。
这种模式使我们的实现灵活且易于从一种技术迁移到另一种技术。
例如,如果我们要使用iBatis ORM框架,我们所需要做的就是为iBatis提供DAO实现,然后更改spring bean配置文件。

在上面的示例中,我正在使用Hibernate会话事务管理。
但是我们也可以使用带有@Transactional注释的Spring声明式事务管理,更多内容请参见Spring Transaction Management。

用于Hibernate 3集成的Spring Bean配置文件

首先让我们看一下Hibernate 3集成所需的spring bean配置,稍后我们将详细研究。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="https://www.springframework.org/schema/beans"
	xmlns:xsi="https://www.w3.org/2001/XMLSchema-instance" xmlns:aop="https://www.springframework.org/schema/aop"
	xmlns:tx="https://www.springframework.org/schema/tx"
	xsi:schemaLocation="https://www.springframework.org/schema/beans https://www.springframework.org/schema/beans/spring-beans.xsd
		https://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop-4.0.xsd
		https://www.springframework.org/schema/tx https://www.springframework.org/schema/tx/spring-tx-4.0.xsd">

	<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
		destroy-method="close">
		<property name="driverClassName" value="com.mysql.jdbc.Driver" 
		<property name="url" value="jdbc:mysql://localhost:3306/TestDB" 
		<property name="username" value="hyman" 
		<property name="password" value="hyman123" 
	</bean>

<!-- Hibernate 3 XML SessionFactory Bean definition-->
<!-- 	<bean id="hibernate3SessionFactory"
		class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
		<property name="dataSource" ref="dataSource" 
		<property name="mappingResources">
			<list>
				<value>person.hbm.xml</value>
			</list>
		</property>
		<property name="hibernateProperties">
			<value>
				hibernate.dialect=org.hibernate.dialect.MySQLDialect
			</value>
		</property>
	</bean> -->

<!-- Hibernate 3 Annotation SessionFactory Bean definition-->
	<bean id="hibernate3AnnotatedSessionFactory"
		class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
		<property name="dataSource" ref="dataSource" 
		<property name="annotatedClasses">
			<list>
				<value>com.theitroad.model.Person</value>
			</list>
		</property>
		<property name="hibernateProperties">
			<props>
				<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
				<prop key="hibernate.current_session_context_class">thread</prop>
				<prop key="hibernate.show_sql">false</prop>
			</props>
		</property>
	</bean>
	
	<bean id="personDAO" class="com.theitroad.dao.PersonDAOImpl">
		<property name="sessionFactory" ref="hibernate3AnnotatedSessionFactory" 
	</bean>
</beans>

我们可以通过两种方式向Hibernate提供数据库连接详细信息,首先是将所有内容传递给" hibernateProperties",其次是创建一个DataSource,然后将其传递给休眠状态。
我更喜欢第二种方法,这就是为什么我们有Apache Commons DBCP依赖项来通过设置数据库连接属性来创建" BasicDataSource"的原因。

对于Spring和Hibernate 3集成,Spring ORM提供了两个类–休眠映射基于XML时的org.springframework.orm.hibernate3.LocalSessionFactoryBean和基于注释的映射的org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean。
如果您使用的是基于XML的映射,则在注释中提供了LocalSessionFactoryBean的简单bean配置。
AnnotationSessionFactoryBean扩展了LocalSessionFactoryBean类,因此它具有休眠集成的所有基本属性。

这些属性是可以自我理解的,并且大多与休眠有关,因此我将不对其进行详细介绍。
但是,如果您想知道hibernateProperties,annotatedClasses来自何处,则需要研究bean类的源代码。

请注意personDAO的bean定义,就像我之前说的,如果我们必须切换到其他ORM框架,则需要在此处更改实现类并设置所需的任何其他属性。

Spring 4 Hibernate 3测试程序

现在我们的设置已经准备就绪,让我们编写一个简单的程序来测试我们的应用程序。

package com.theitroad.main;

import java.util.List;

import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.theitroad.dao.PersonDAO;
import com.theitroad.model.Person;

public class SpringHibernateMain {

	public static void main(String[] args) {

		ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
		
		PersonDAO personDAO = context.getBean(PersonDAO.class);
		
		Person person = new Person();
		person.setName("hyman"); person.setCountry("San Franceco");
		
		personDAO.save(person);
		
		System.out.println("Person::"+person);
		
		List<Person> list = personDAO.list();
		
		for(Person p : list){
			System.out.println("Person List::"+p);
		}
		//close resources
		context.close();	
	}
}

当我们执行上述程序时,由于没有正确设置日志记录,因此会得到很多与Hibernate相关的输出,但这超出了本教程的范围。
但是,我们得到了程序生成的以下输出。

Person::id=3, name=hyman, country=San Franceco
Person List::id=1, name=hyman, country=San Franceco
Person List::id=2, name=hyman, country=San Franceco
Person List::id=3, name=hyman, country=San Franceco

Spring 4 Hibernate 4集成更改

现在,让我们将应用程序更改为使用Hibernate 4而不是Hibernate3。
对于此迁移,我们只需要进行以下配置更改。

  • 将pom.xml文件中的休眠版本更改为4.3.5.Final,如上面的注释所示。

  • 更改spring bean配置文件,到现在为止,您必须已经知道Spring bean配置文件是spring和hibernate框架集成的关键。
    下面的spring bean配置文件将适用于Spring 4和Hibernate 4版本。

对于hibernate 4,我们需要为SessionFactory bean使用org.springframework.orm.hibernate4.LocalSessionFactoryBean,Spring ORM合并了Hibernate 3的两个类,现在有一个类,这样可以避免混淆。
所有其他配置与以前相同。