• Hibernate 处理事务


    1. Hibernate 的持久化类

    1.1 什么是持久化类

    • 持久化类: 就是一个 Java 类(JavaBean),这个 Java类与表建立了映射关系就可以是持久化类;
    • 持久化类 = JavaBean + xxx.hbm.xml;

    1.2 持久化类的编写规则

    • 提供一个无参数的构造方法,因为底层需要进行反射;
    • 提供一个唯一的标识 OID, 即映射数据表主键;
      数据库中通过主键, Java 对象通过地址确定对象. 持久化类通过唯一标识 OID 确定记录.
    • 所有属性提供公有的 set 或者 get 方法;
    • 标识属性应尽量使用基本数据类型的包装类;

    1.3 自然主键和代理主键

    • 自然主键: 对象本身的一个属性. 例如,创建一个人员表,使用身份证号(唯一的)作为表的主键.
    • 代理主键: 不是对象本身的一个属性. 例如,创建一个人员表,为每个人员单独创建一个作为主键的字段.
    • 创建表时,尽量使用代理主键.

    1.4 主键生成的策略

    <id name="cust_id" column="cust_id">
        // 主键的生成策略
        <generator class="native"/>
    </id>
    
    1. increment: 适用于 short, int, long 作为主键,不是使用的数据库自动增长机制;
      • Hibernate 中提供的一种增长机制;
      • 先进行查询: select max(id) from user, 再进行插入,将查询出的最大值+1 作为新记录的主键;
      • 缺点:不能在集群环境下或者有并发访问的情况下使用;
    2. identity: 适用于 short, int, long 作为主键,但是这个必须使用在自动增长的数据库中.
      • 底层使用的是数据库的自动增长机制;
    3. sequence: 适用于short, int, long 作为主键.
      • 底层使用的是序列的增长方式,例如 Oracle 数据库.
    4. uuid: 适用于 char, varchar 类型的作为主键;
      • 使用随机的字符串作为主键;
    5. native: 使用本地策略,根据底层的数据库不同,自动选择适用于该种数据库的生成策略.
      • 如果底层使用的 MySql 数据库,相当于 identity;
      • 如果底层使用的是 Oracle 数据库,相当于 sequence;
    6. assigned: 主键的生成不用 Hibernate 管理,必须手动设置主键;

    2. Hibernate 持久化对象的状态

    2.1 Hibernate 持久化类的状态

    1. Hibernate 为了管理持久化类,将持久化类分成了三个状态:
      • 瞬时态(Transient Object), 没有持久化标识 OID,没有被纳入到 Session 对象的管理;
      • 持久态(Persistent Object)
        • 有持久化标识OID,已经被纳入到 Session 对象的管理;
        • 注意: 持久化持久态的对象有自动更新数据库的能力!! 因为 Session 的一级缓存.
      • 托管态(Detached Object), 有持久化标识 OID,没有被纳入到 Session 对象的管理;

    2.2 Hibernate 持久化对象的状态转换

    1. 瞬时态
      • 获得瞬时态对象: User user = new User();
      • 瞬时态对象转换成持久态: save() 或 saveOrUpdate();
      • 瞬时态对象转换成托管态(不推荐): user.setId();
    2. 持久态
      • 获得持久态的对象: get() 或 load();
      • 持久态转换成瞬时态对象: delete();
      • 持久态对象转成托管态对象: session 的 close() 或 evict() 或 clear()
    3. 托管态
      • 托管态转换成瞬时态: user.setId(null);
      • 托管态转换成持久态: update() 或 saveOrUpdate();

    3. Hibernate 的一级缓存

    3.1 什么是缓存

    • 缓存其实就是一块内存空间,将数据源(数据库或者文件)中的数据存放到缓存中,再次获取的时候,直接从缓存中获取.
      可以提升程序的性能.

    3.2 Hibernate 框架提供了两种缓存

    1. 一级缓存

      • 自带的,不可卸载的;
      • 一级缓存的生命周期与 session 一致, 一级缓存称为 session 级别的缓存;
    2. 二级缓存

      • 二级缓存可以在多个 session 中共享数据;
      • 二级缓存称为 sessionFactory 级别的缓存;
      • 二级缓存是为了增强一级缓存,一级缓存的生命周期比较短暂;
      • 二级缓存默认没有开启,需要手动配置才可以使用;
    3. session 对象的缓存概述

      • Session 对象中有一系列 java 的集合,这些集合构成了一级缓存;
    4. Session 中与一级缓存相关的方法

      • session.clear(): 清空一级缓存;
      • session.evict(Object entity): 从一级缓存中清除指定的实体对象;
      • session.flush(): 刷出缓存;
    5. Hibernate 框架是如何做到数据发生变化时,进行同步操作的呢?

    4. Hibernate 中的事务和并发

    4.1 Hibernate 框架中设置隔离级别

    1. 需要在 hibernate.cfg.xml 的配置文件中通过标签来配置;
      • <property name="hibernate.connection.isolation">4</property>
      • "1"表示 Read uncommitted isolation
      • "2"表示 Read committed isolation
      • "4"表示 Repeatable read isolation
      • "8"表示 Serializable isolation

    4.2 丢失更新的问题

    1. 如果不考虑隔离性,也会产生写入数据的问题,即丢失更新的问题;

    2. 例如: A 和 B 两个事务同时对某一条记录做修改,就会引发丢失更新的问题;

    3. 解决方案

      • "悲观锁"
        • 采用的是数据库提供的一种锁机制,如果采用了这种机制,在SQL语句的后面添加for update子句
        • 当 A 事务在操作该条记录时,会把该条记录锁起来,其他事务是不能操作这条记录的;
        • 只有当 A 事务提交后,锁释放了,其他事务才能操作该条记录;
      • "乐观锁"
        • 采用版本号的机制来解决,会在表结构添加一个字段 version=0,默认值为 0;
        • 当 A 事务在操作完该条记录,提交事务时,会先检查版本号,只有版本号值相同,才可以提交事务.
          同时,更新版本号 version=1;
        • 当 B 事务在操作完该条记录,提交事务时,会先检查版本号,如果发现版本号不同,程序会抛出异常.
    4. Hibernate 框架解决丢失更新的问题

      • "悲观锁":较少使用,效率慢
      • "乐观锁"
        • 在对应的 JavaBean 中添加一个属性,名称可以是任意的.并提供 get 和 set 方法.
          例如:private Integer version;
        • 在映射的配置文件中,提供 <version name="version"/> 标签即可;

    4.3 绑定本地 Session

    1. JavaWeb 中的事务,需要在业务层使用 Connection 来开启事务
      • 一种是通过参数的方式传递;
      • 另一种是把 Connection 绑定到 ThreadLocal 对象中;
    2. Hibernate 框架,使用 session 对象开启事务.框架提供了 ThreadLocal 的方式,传递 session 对象

    // 需要在 hibernate.cfg.xml 的配置文件中提供如下配置
        <property name="hibernate.current_session_context_class">thread</property>
    
    // 重写 HiberanteUtils 工具类
        public class HibernateUtils {
            private static final Configuration CONFIG;
            private static final SessionFactory FACTORY;
    
            static{
                CONFIG = new Configuration().configure();
                FACTORY = CONFIG.buildSessionFactory();
            }
    
            public static Session getCurrentSession(){
    
                // 从 ThreadLocal 中获取当前 session 对象
                // 该对象不用再手动关闭,线程结束了,会自动关闭.
                return FACTORY.getCurrentSession();
            }
        }
    

    参考资料

  • 相关阅读:
    Volatile的作用---http://www.cnblogs.com/xing901022/p/7840684.html
    基于JDBC持久化的事务管理-https://www.cnblogs.com/xing901022/p/4272420.html
    Class的isAssignableFrom方法--其他博主的博客
    深入并发二 ThreadLocal源码与内存泄漏相关分析 https://www.cnblogs.com/qmlingxin/p/9412061.html
    Beta阶段项目总结
    Alpha阶段项目总结
    Alpha版总结会议——班级派
    第二冲刺阶段——站立会议第十四天6月7日
    第二冲刺阶段——站立会议第十三天6月6日
    第二冲刺阶段——站立会议第十二天6月5日
  • 原文地址:https://www.cnblogs.com/linkworld/p/7697697.html
Copyright © 2020-2023  润新知