上篇文章(传送门:Hibernate学习笔记_01)介绍了Hibernate是什么,如何搭建,配置文件详解以及Hibernate的一些基本API详解这几个方面做了简单介绍,那么本文将会从一下5个方面记录Hibernate的学习经历:
1.hibernate中的实体规则
2.hibernate中的对象状态
3.hibernate进阶—— 一级缓存
4.hibernate中的事务
5.hibernate中的批量查询(概述)
Ⅰ.Hibernate中的实体规则
在Hibernate使用中,需要创建与数据库表对应的实体,并在映射文件中配置.在创建实体的时候需要注意一些细节.
(1)实体类创建时需要注意5个事项:
1.持久化类提供无参数构造
2.成员变量私有,提供共有get/set方法访问.需提供属性
3.持久化类中的属性,应尽量使用包装类型
4.持久化类需要提供oid.与数据库中的主键列对应
5.不要用final修饰class(hibernate使用cglib代理生成代理对象.代理对象是继承被代理对象.如果被final修饰.将无法生成代理.)
(2)主键类型
自然主键(少见):表的业务列中,有某业务列符合,必须有,并且不重复的特征时,该列可以作为主键使用.
代理主键(常见):表的业务列中,没有某业务列符合,必须有,并且不重复的特征时,创建一个没有业务意义的列作为主键.
(3)主键生成策略
代理主键
identity : 主键自增.由数据库来维护主键值.录入时不需要指定主键.
sequence: Oracle中的主键生成策略.
increment: 主键自增.由hibernate来维护.每次插入前会先查询表中id最大值.+1作为新主键值.
hilo: 高低位算法.主键自增.由hibernate来维护.开发时不使用.
native:hilo+sequence+identity 自动三选一策略.
uuid: 产生随机字符串作为主键. 主键类型必须为string 类型.
在实际的开发过程中可以根据需要来选择合适的主键生成策略,一般来说比叫常见的是使用native,通常是在实体xxx.hbn.xml中<generator class="native"></generator>属性中配置
自然主键
assigned:自然主键生成策略. hibernate不会管理主键值.由开发人员自己录入.
Ⅱ.Hibernate中对象的状态
在Hibernate中,对象通常分为三种状态:
1.瞬时状态
没有id,没有在session缓存中
1 public void fun1() { 2 // 1.获得session 3 Session session = HibernateUtils.openSession(); 4 // 2.控制事务 5 Transaction tx = session.beginTransaction(); 6 // 3.执行操作 7 Customer c = new Customer(); // 没有id,没有在session缓存中 ==> 瞬时状态 8 9 c.setCust_name("惠普"); // 瞬时状态 10 11 session.save(c); // 持久化状态,有id,在session缓存中 12 // 4.提交事务 13 tx.commit(); 14 // 5.关闭资源 15 session.close(); // 游离|脱管状态 ,有id,没有在session缓存中 16 }
2.持久化状态
有id,在session缓存中
持久化状态的特点: 持久化状态的对象的任何变化都会同步到数据库当中.
3.游离|托管状态
有id,没有在session缓存中
在开发操作过程中,这三种状态是可以互相转变的
Ⅲ.Hibernate进阶——一级缓存
通常情况下,缓存是为了提高效率,在大家最常见的电脑中,CPU也存在缓存,硬盘也存在缓存,内存中同样也存在缓存,而hibernate中的一级缓存同样也是为了提高操作数据库的效率.
Hibernate中缓存提高效率的手段有两种:
1.提高查询效率
2.减少不必要的修改语句的发送
Ⅳ.Hibernate中的事务
大家都知道,事务有四大特性()ACID)分别是:
原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)
事务并发带来的问题有:1.脏读、2.不可重复读、3.幻|虚读。
事务的隔离级别有四种情况:(后面的数字分别代表上面并发带来的问题)
读未提交- 123
读已提交 - 23
可重复读(mysql默认级别)-3
串行化 - 没有问题
那么如何在HIbernate中指定数据库的隔离级别呢?只要在Hibernate.cfg.xml文件中增加如下一行配置即可
<!-- 指定Hibernate操作数据库时的隔离级别 ## specify a JDBC isolation level #hibernate.connection.isolation 1|2|4|8 0001 1 读未提交 0010 2 读已提交 0100 4 可重复读 1000 8 串行化 --> <property name="hibernate.connection.isolation">4</property>
那么我们在项目中又该如何去管理事务呢?
业务开始之前打开事务,业务执行之后提交事务. 执行过程中出现异常.回滚事务.
在dao层操作数据库需要用到session对象.在service控制事务也是使用session对象完成. 我们要确保dao层和service层使用的使用同一个session对象
在hibernate中,确保使用同一个session的问题,hibernate已经帮我们解决了. 我们开发人员只需要调用sf.getCurrentSession()方法即可获得与当前线程绑定的session对象
注意1: 调用getCurrentSession方法必须配合主配置中的一段配置
<!-- 指定session与当前线程绑定 --> <property name="hibernate.current_session_context_class">thread</property>
注意2:通过getCurrentSession方法获得的session对象.当事务提交时,session会自动关闭.不要手动调用close关闭.
在Service层
Dao层