Hibernate与Spring整合实践实例

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS PostgreSQL,集群系列 2核4GB
简介: Hibernate与Spring整合实践实例

spring与hibernate的整合目的就是为了让 IOC 容器来管理 Hibernate 的核心接口SessionFactory以及让 Hibernate 使用上 Spring 的声明式事务来进行事务操作.


【1】Hibernate使用XML方式与Spring整合

即,Model上面不添加注解,使用*.hbm.xml方式与数据库进行关联。

① applicationContext.xm:

<?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:aop="http://www.springframework.org/schema/aop"
  xmlns:context="http://www.springframework.org/schema/context"
  xmlns:tx="http://www.springframework.org/schema/tx"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
    http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">
  <!-- 导入资源文件 -->
  <context:property-placeholder location="classpath:db.properties"/>
  <context:component-scan base-package="com"></context:component-scan>
  <!-- 配置 C3P0 数据源 -->
  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    <property name="user" value="${jdbc.user}"></property>
    <property name="password" value="${jdbc.password}"></property>
    <property name="driverClass" value="${jdbc.driverClass}"></property>
    <property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>  
    <property name="initialPoolSize" value="${jdbc.initPoolSize}"></property>
    <property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property>
  </bean>
  <!-- 配置 SessionFactory -->
  <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
    <property name="dataSource" ref="dataSource"></property>
    <property name="configLocation" value="classpath:hibernate.cfg.xml"></property>
    <!-- 如果不想使用额外的hibernate.cfg.xml,则需要进行如下设置 -->
    <property name="hibernateProperties">
            <props>  
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</prop>  
                <prop key="hibernate.hbm2ddl.auto">update</prop>  
                <prop key="hibernate.show_sql">true</prop>  
                <prop key="hibernate.format_sql">true</prop>  
            </props>  
        </property>
    <property name="mappingLocations" value="classpath:com/jane/ssh/entities/*.hbm.xml"></property>
  </bean>
  <!-- 这里配置与MyBatis整合Spring很类似 -->
  <!-- 配置 Spring 的声明式事务 -->
  <!-- 1. 配置 hibernate 的事务管理器 -->
  <bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
    <property name="sessionFactory" ref="sessionFactory"></property>
  </bean>
  <!-- 2. 配置事务属性 -->
   <!-- 配置哪些方法要加入事务控制 -->
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <!-- 让所有的方法都加入事务管理,为了提高效率,可以把一些查询之类的方法设置为只读事务 -->
            <tx:method name="*" propagation="REQUIRED" read-only="true"/>
            <!-- 以下方法都是可能涉及写改的方法,就无法设置为只读 -->
            <tx:method name="add*" propagation="REQUIRED"/>
            <tx:method name="del*" propagation="REQUIRED"/>
            <tx:method name="update*" propagation="REQUIRED"/>
            <tx:method name="save*" propagation="REQUIRED"/>
        </tx:attributes>
    </tx:advice>
  <!-- 3. 配置事务切入点, 再把事务属性和事务切入点关联起来 -->
  <aop:config>
    <aop:pointcut expression="execution(* com.jane.ssh.service.*.*(..))" id="txPointcut"/>
    <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
  </aop:config>
  <!-- 开启HibernateTemplate,并且为其注入SessionFactory
    使用HibernateTemplate不太方便的就是要获取session得通过getSessionFactory()方法获取
    注意:这里之所以配置这个是因为Dao继承了HibernateDaoSupport
  -->
    <bean id="hibernateTemplate" class="org.springframework.orm.hibernate4.HibernateTemplate">
        <property name="sessionFactory" ref="sessionFactory"/>
    </bean>
</beans>

② hibernate.cfg.xml

由于数据源,*.hbm.xml都在applicationContext.xml中进行了配置,故而hibernate.cfg.xml极其简化:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
    "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <session-factory>
      <!-- 配置 hibernate 的基本属性 -->
      <!-- 方言 -->
      <property name="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
      <!-- 是否显示及格式化 SQL -->
      <property name="hibernate.show_sql">true</property>
      <property name="hibernate.format_sql">true</property>
      <!-- 生成数据表的策略 -->
      <property name="hibernate.hbm2ddl.auto">update</property>
      <!-- 启用二级缓存 -->
    <property name="cache.use_second_level_cache">true</property>
      <!-- 配置使用的二级缓存的产品 -->
      <property name="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</property>
      <!-- 配置启用查询缓存 -->
      <property name="cache.use_query_cache">true</property>
    </session-factory>
</hibernate-configuration>

【2】Hibernate注解版与Spring整合

即,Model上面添加注解,此时不再存在*.hbm.xml。


① applicationContext.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"
  xmlns:aop="http://www.springframework.org/schema/aop"
  xmlns:context="http://www.springframework.org/schema/context"
  xmlns:tx="http://www.springframework.org/schema/tx"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
    http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">
  <!-- 导入资源文件 -->
  <context:property-placeholder location="classpath:db.properties"/>
  <context:component-scan base-package="com"></context:component-scan>
  <!-- 配置 C3P0 数据源 -->
  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    <property name="user" value="${jdbc.user}"></property>
    <property name="password" value="${jdbc.password}"></property>
    <property name="driverClass" value="${jdbc.driverClass}"></property>
    <property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>  
    <property name="initialPoolSize" value="${jdbc.initPoolSize}"></property>
    <property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property>
  </bean>
  <!-- 配置 SessionFactory -->
  <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
    <property name="dataSource" ref="dataSource"></property>
    <property name="hibernateProperties">
            <props>  
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</prop>  
                <prop key="hibernate.hbm2ddl.auto">update</prop>  
                <prop key="hibernate.show_sql">true</prop>  
                <prop key="hibernate.format_sql">true</prop>  
            </props>  
        </property>
    <!-- 如果实体类使用了注解,通常会使用如下方式
    通过扫描包方式设置Spring取哪个包查找相应的实体类 -->
        <property name="packagesToScan">
            <value>com.jane.ssh.entities</value>
        </property>
        -->
  </bean>
  <!-- 配置 Spring 的声明式事务 -->
  <!-- 1. 配置 hibernate 的事务管理器 -->
  <bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
    <property name="sessionFactory" ref="sessionFactory"></property>
  </bean>
  <!-- 2. 配置事务属性 -->
   <!-- 配置哪些方法要加入事务控制 -->
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <!-- 让所有的方法都加入事务管理,为了提高效率,可以把一些查询之类的方法设置为只读事务 -->
            <tx:method name="*" propagation="REQUIRED" read-only="true"/>
            <!-- 以下方法都是可能涉及写改的方法,就无法设置为只读 -->
            <tx:method name="add*" propagation="REQUIRED"/>
            <tx:method name="del*" propagation="REQUIRED"/>
            <tx:method name="update*" propagation="REQUIRED"/>
            <tx:method name="save*" propagation="REQUIRED"/>
        </tx:attributes>
    </tx:advice>
  <!-- 3. 配置事务切入点, 再把事务属性和事务切入点关联起来 -->
  <aop:config>
    <aop:pointcut expression="execution(* com.jane.ssh.service.*.*(..))" id="txPointcut"/>
    <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
  </aop:config>
  <!-- 开启HibernateTemplate,并且为其注入SessionFactory
    使用HibernateTemplate不太方便的就是要获取session得通过getSessionFactory()方法获取
    注意:这里之所以配置这个是因为Dao继承了HibernateDaoSupport
  -->
    <bean id="hibernateTemplate" class="org.springframework.orm.hibernate4.HibernateTemplate">
        <property name="sessionFactory" ref="sessionFactory"/>
    </bean>
</beans>

② hibernate.cfg.xml

由于数据源,*.hbm.xml都在applicationContext.xml中进行了配置,故而hibernate.cfg.xml极其简化:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
    "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <session-factory>
      <!-- 配置 hibernate 的基本属性 -->
      <!-- 方言 -->
      <property name="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
      <!-- 是否显示及格式化 SQL -->
      <property name="hibernate.show_sql">true</property>
      <property name="hibernate.format_sql">true</property>
      <!-- 生成数据表的策略 -->
      <property name="hibernate.hbm2ddl.auto">update</property>
      <!-- 启用二级缓存 -->
    <property name="cache.use_second_level_cache">true</property>
      <!-- 配置使用的二级缓存的产品 -->
      <property name="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</property>
      <!-- 配置启用查询缓存 -->
      <property name="cache.use_query_cache">true</property>
    </session-factory>
</hibernate-configuration>

当然,通常情况下,会去掉hibernate.cfg.xml,而将Hibernate相关属性配置在sessionFactory中。


③ 常见的Hibernate注解示例

package domain;
import java.io.Serializable;
import java.util.Date;
import java.util.LinkedList;
import java.util.List;
import org.hibernate.annotations.Cache;
import org.hibernate.annotations.CacheConcurrencyStrategy;
import javax.persistence.*;
@Entity
@Table(name="article_inf")
@Cache(usage=CacheConcurrencyStrategy.READ_WRITE)
public class Article implements Serializable{
  private static final long serialVersionUID = 48L;
  @Id @Column(name="article_id")
  @GeneratedValue(strategy=GenerationType.IDENTITY)
  private Integer id;
  @ManyToOne(targetEntity=Login.class)
  @JoinColumn(name="user_id",referencedColumnName="user_id",nullable=false)
  private Login login;
  @Column(name="boss_id")
  private Integer bossId;
  @Temporal(TemporalType.TIMESTAMP)
  @Column(name="create_date")
  private Date createDate;
  @Temporal(TemporalType.TIMESTAMP)
  @Column(name="modify_date")
  private Date modifyDate;
  @Column(name="article_authority")
  private Integer authority;
  @Column(name="title")
  private String title;
  @Column
  private String writer;
  @ElementCollection(targetClass=String.class)
  @CollectionTable(name="art_content",joinColumns=@JoinColumn(name="article_id",nullable=false))
  @Column(name="content",length=5000) @OrderColumn(name="No_")
  private List<String> contents=new LinkedList<>();
  @OneToMany(targetEntity=ArticlePictures.class,orphanRemoval=true,mappedBy="article",cascade=CascadeType.ALL)
  private List<ArticlePictures> pictures=new LinkedList<>();
  @OneToMany(targetEntity=ArticleComments.class,orphanRemoval=true,mappedBy="article",cascade=CascadeType.ALL)
  private List<ArticleComments> comments=new LinkedList<>();
  @OneToMany(targetEntity=ArticleVisitors.class,orphanRemoval=true,mappedBy="article",cascade=CascadeType.ALL)
  private List<ArticleVisitors> visitors=new LinkedList<>();
  @OneToMany(targetEntity=ArticleCharm.class,orphanRemoval=true,mappedBy="article",cascade=CascadeType.ALL)
  private List<ArticleCharm> charm=new LinkedList<>();
  //id
  public void setId(Integer id){
    this.id=id;
  }
  public Integer getId(){
    return id;
  }
  //login
  public void setLogin(Login login){
    this.login=login;
  }
  public Login getLogin(){
    return login;
  }
  //bossId
  public void setBossId(Integer bid){
    this.bossId=bid;
  }
  public Integer getBossId(){
    return bossId;
  }
  //createDate
  public void setCreateDate(Date date){
    this.createDate=date;
  }
  public Date getCreateDate(){
    return createDate;
  }
  //modifyDate
  public void setModifyDate(Date date){
    this.modifyDate=date;
  }
  public Date getModifyDate(){
    return modifyDate;
  }
  //authority
  public void setAuthority(Integer author){
    this.authority=author;
  }
  public Integer getAuthority(){
    return authority;
  }
  //title
  public void setTitle(String title){
    this.title=title;
  }
  public String getTitle(){
    return title;
  }
  //writer
  public void setWriter(String writer){
    this.writer=writer;
  }
  public String getWriter(){
    return writer;
  }
  //contents
  public void setContents(List<String> con){
    this.contents=con;
  }
  public List<String> getContents(){
    return contents;
  }
  //pictures
  public void setPictures(List<ArticlePictures> pic){
    this.pictures=pic;
  }
  public List<ArticlePictures> getPictures(){
    return pictures;
  }
  //comments
  public void setComments(List<ArticleComments> coms){
    this.comments=coms;
  }
  public List<ArticleComments> getComments(){
    return comments;
  }
  //visitors
  public void setVisitors(List<ArticleVisitors> visitors){
    this.visitors=visitors;
  }
  public List<ArticleVisitors> getVisitors(){
    return visitors;
  }
  //charm
  public void setCharm(List<ArticleCharm> charm){
    this.charm=charm;
  }
  public List<ArticleCharm> getCharm(){
    return charm;
  }
}

至于Hibernate注解有哪些,注解如何使用,请参考系列博文:Spring Data JPA那些事。毕竟,JPA的默认实现就是Hibernate。

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
9天前
|
XML Java 数据格式
🌱 深入Spring的心脏:Bean配置的艺术与实践 🌟
本文深入探讨了Spring框架中Bean配置的奥秘,从基本概念到XML配置文件的使用,再到静态工厂方式实例化Bean的详细步骤,通过实际代码示例帮助读者更好地理解和应用Spring的Bean配置。希望对你的Spring开发之旅有所助益。
59 3
|
25天前
|
XML Java 数据格式
Spring Core核心类库的功能与应用实践分析
【12月更文挑战第1天】大家好,今天我们来聊聊Spring Core这个强大的核心类库。Spring Core作为Spring框架的基础,提供了控制反转(IOC)和依赖注入(DI)等核心功能,以及企业级功能,如JNDI和定时任务等。通过本文,我们将从概述、功能点、背景、业务点、底层原理等多个方面深入剖析Spring Core,并通过多个Java示例展示其应用实践,同时指出对应实践的优缺点。
53 14
|
21天前
|
缓存 Java 数据库连接
Spring框架中的事件机制:深入理解与实践
Spring框架是一个广泛使用的Java企业级应用框架,提供了依赖注入、面向切面编程(AOP)、事务管理、Web应用程序开发等一系列功能。在Spring框架中,事件机制是一种重要的通信方式,它允许不同组件之间进行松耦合的通信,提高了应用程序的可维护性和可扩展性。本文将深入探讨Spring框架中的事件机制,包括不同类型的事件、底层原理、应用实践以及优缺点。
49 8
|
26天前
|
负载均衡 Java 开发者
深入探索Spring Cloud与Spring Boot:构建微服务架构的实践经验
深入探索Spring Cloud与Spring Boot:构建微服务架构的实践经验
88 5
|
23天前
|
XML 前端开发 安全
Spring MVC:深入理解与应用实践
Spring MVC是Spring框架提供的一个用于构建Web应用程序的Model-View-Controller(MVC)实现。它通过分离业务逻辑、数据、显示来组织代码,使得Web应用程序的开发变得更加简洁和高效。本文将从概述、功能点、背景、业务点、底层原理等多个方面深入剖析Spring MVC,并通过多个Java示例展示其应用实践,同时指出对应实践的优缺点。
53 2
|
27天前
|
安全 Java 数据安全/隐私保护
如何使用Spring Boot进行表单登录身份验证:从基础到实践
如何使用Spring Boot进行表单登录身份验证:从基础到实践
45 5
|
27天前
|
监控 Java 数据安全/隐私保护
如何用Spring Boot实现拦截器:从入门到实践
如何用Spring Boot实现拦截器:从入门到实践
48 5
|
1月前
|
SQL Java 数据库连接
从理论到实践:Hibernate与JPA在Java项目中的实际应用
本文介绍了Java持久层框架Hibernate和JPA的基本概念及其在具体项目中的应用。通过一个在线书店系统的实例,展示了如何使用@Entity注解定义实体类、通过Spring Data JPA定义仓库接口、在服务层调用方法进行数据库操作,以及使用JPQL编写自定义查询和管理事务。这些技术不仅简化了数据库操作,还显著提升了开发效率。
45 3
|
1月前
|
Java 测试技术 数据库连接
使用Spring Boot编写测试用例:实践与最佳实践
使用Spring Boot编写测试用例:实践与最佳实践
70 0
|
1月前
|
数据采集 Java 数据安全/隐私保护
Spring Boot 3.3中的优雅实践:全局数据绑定与预处理
【10月更文挑战第22天】 在Spring Boot应用中,`@ControllerAdvice`是一个强大的工具,它允许我们在单个位置处理多个控制器的跨切面关注点,如全局数据绑定和预处理。这种方式可以大大减少重复代码,提高开发效率。本文将探讨如何在Spring Boot 3.3中使用`@ControllerAdvice`来实现全局数据绑定与预处理。
69 2