从之前的代码中我们可以看到,Hibernate中的事物一般是由s.beginTranscation()开始,由s.getTranscation().commit()结束,这个事务一旦开启,无论你中间进行了什么操作多少操作,这些操作都在一个事务中,如果其中任意一个失败了,那么所有的操作都是失败的,因为这个错误没有导致事务运行成功。这是什么情况?
因为数据库事务必须具备ACID特性:
- 原子性:这个事务不可分割,要么全部成功,要么全部失败
- 一致性:事务不能破坏关系数据的完整性和业务逻辑的一致性,比如转账,在这个流程结束后你必须保证两个账户的综合和转账之前是一致的,不能因为转账之间出错导致其中一个用户金额少了而另一个没有增加。
- 隔离性:多个数据库同时操作相同数据时候,每个事物都有各自完整的数据空间。
- 持久性:只要事物成功的结束,那么对数据库的更新就必须永久的保存下来,即使系统发生崩溃,重启数据库后,数据库还能恢复到事务成功结束时的状态。
数据库支持两种事务模式:
自动提交模式:每个SQL语句都是一个独立的事务,数据库执行完一条SQL语句后,会自动提交事务。
手动提交模式:由数据库的客户程序显式指定事务的开始和结束边界。
那么,什么是事务的开始和结束边界呢?
只要声明一个事务,数据库就会自动保证数据库的ACID特性。
声明事务包含:
事务的开始边界
事务的正常结束边界:提交事务,永久保存。
事务的异常结束边界:撤销事务,数据库回退到执行事务前的状态。
--------------------------------------------JDBC Connection类的事务控制方法--------------------------
setAutoCommmit(boolean autocommit)设置事务是否自动提交,默认为自动提交事务。
commit()提交事务。
rollback()回退事务。
-----------------------------------Hibernate控制事务的方法---------------------------------------
1.调用SessionFactory不带参数的openSession方法,从连接池获得连接,Session自动将连接设置为手动提交事务模式。
SessionFactory sf=new Configuration().configura().buildSessionFactory();
Session s=sf.openSession();
如果调用带connection参数的openSession()方法,则自己需要设置手动提交。2.声明事务的开启
SessionFactory sf=new Configuration().configure().buildSessionFactory();
Session s=sf.openSession();
s.beginTransaction();
3.提交事务:s.getTransaction().commit();
s.getTransaction().rollBack();
其实一个Session可以应对多个事务,但是任何时候一个session只允许有一个未提交的事务,不能同时开启两个事务。而且不管事务的成功与否,都应该最后调用session的close()方法关闭session。
------------------------------------------------------------Session------------------------------------------------
Hibernate获取Session有两种方式,分别是openSession()和getCurrentSession()
但是这两个有什么区别?
- openSession()每次都会得到一个新的Session对象,getCurrentSession在同一个线程中,每次都会获取到相同的session对象,但是在不同的线程中获取的session对象不同。
- openSession()只有在增加、删除和修改的时候需要事务,而查询是不需要开启事务的。getCurrentSession是所有的操作都必须开启一个事务在事务中进行,并且事务提交后,session就会自动关闭,不需要再显示关闭。