事务(一)

来源:互联网 发布:航天科工33所 知乎 编辑:程序博客网 时间:2024/05/16 15:27
背景:
跳槽了,在找工作,这一周中,很多面试官都问了我关于事务的问题,我个人觉得答得不是很好,于是今天查下资料做个总结(可能不全)


本文大多内容来源自别人的博客和百度:如http://lovetocool.iteye.com/blog/1132317


1.什么是事务:
按照百度百科的说明:
I,事务:一般是指要做的或所做的事情。
II,事务:(计算机术语中)是访问并可能更新数据库中各种数据项的一个程序执行单元(unit)。


事务通常由高级数据库操纵语言或编程语言(如SQL,Java)书写的用户程序的执行所引起,并用形如begin transaction和end transaction语句(或函数调用)来界定。事务由事务开始(begin transaction)和事务结束(end transaction)之间执行的全体操作组成。


2.事务的特性(ACID):
A(Atomicity)原子性:
事务是一个完整的操作。事务的各步操作是不可分的(原子的);要么都执行,要么都不执行,只能二选一.
C(Consistency)一致性:
当事务完成时,数据必须处于一致状态(数据库中的所有数据和现实保持一致).
I(Isolation)隔离性:
对数据进行修改的所有并发事务是彼此隔离的,这表明事务必须是独立的,它不应以任何方式依赖于或影响其他事务.
D(Durability)永久性:
事务完成后,它对数据库的修改被永久保持,事务日志能够保持事务的永久性。


3.事务之间的缺陷(并发问题)
如果存在多个并发事务在运行,而这种事务操作了同一个数据来完成它们的任务,就会导致3个问题的发生.


I,脏读:当一个事务读取了另一个事务尚未提交的更新,就叫脏读取。在另一个事务回滚的情况下,当前事务所读取的另一个事务的数据就是无效的。


II,不可重复读取:在一个事务中执行多次同样的查询操作,但每次查询的结果都不一样,就叫做不可重复读取,通常这种情况是由于数据在二次查询之间被另一个并发的事务所修改。


III,幻读:类似不可重复读取,也是一个事务的更新结果影响到另一个事务问题(它不仅影响另一个事务查询结果,而且还会使查询语句返回一些不同的结果)


这3个问题危害程度依次为:脏读取最大-->不可重复读取-->幻读最小。


4.spring中的事务:
是一系列任务的组成工作单元,在这个工作单元中,所有的任务必须同时执行。它们只有二种可能执行结果,要么所有任务全部执行成功,要么所有任务全部执行失败。 (就跟定义一样)


但是Spring中提供了丰富的事务管理功能,其功能如下:
I,为不同的事务的API提供一致的编程模式 
II,提供更简单,更易地使用的编程式事务管理 
III,支持Spring声明事务 
IV,整合Spring对数据访问的抽像  


4.1.spring中的事务的传播行为:
传播行为是事务应用于方法的边界,它定义了事务的建立,暂停等行为属性
在Spring中共有7种:
PROPAGATION_MANDATORY: 
规定了方法必须在事务中运行,否则会抛出异常 


PROPAGATION_NESTED: 
使方法运行在嵌套事务中,否则这个属性和PROPAGATION_REQUIRED属性的义相同 


PROPAGATION_NEVER 
使当前方法永远不在事务中运行,否则抛出异常 


PROPAGATION_NOT_SUPPORTED 
定义为当前事务不支持的方法,在该方法运行期间正在运行的事务会被暂停 


PROPAGATION_REQUIRED 
规定当前的方法必须在事务中,如果没有事务就创建一个新事务,一个新事务和方法一同开始,随着方法的返回或抛出异常而终止 .


PROPAGATION_REQUIRED_NEW 
当前方法必须创建新的事务来运行,如果现存的事务正在运行就暂停它 


PROPAGATION_SUPPORTS 
规定当前方法支持当前事务处理,但如果没有事务在运行就使用非事务方法执行 


例如:
<tx:advice id="txAdvice">
<tx:attributes>
<!-- 只要是get(无参),find(有参数)打头方法就是只读事务或者不需要事务 -->
<tx:method name="get*" read-only="true" propagation="SUPPORTS" />
<tx:method name="find*" read-only="true" propagation="SUPPORTS" />
<!-- 其他的方法都是使用默认事务管理 -->
<tx:method name="*" />
</tx:attributes>
</tx:advice>


4.2.事务的隔离级别:
为了解决并发事务所带来的三个问题,只有在事务之间建立隔离关系来保证事务的完整性。
ISOLATION_DEFAULT         
使用数据库默认的隔离级别 
    
ISOLATION_COMMITTED     
允许读取其他并发事务已经提交的更新(防此脏读) 
    
ISOLATION_READ_UNCOMMITTED 
允许读取其他并发事务还未提交的更新,会导致事务之间的3个缺陷发生,这是速度最快的一个隔离级别,但同时它的隔离级别也是最低.
    
ISOLATION_REPEATABLE_READ 
除非事务自身修改了数据,否则规定事务多次重复读取数据必须相同(防此脏读,不可重复读) 
   
ISOLATION_SERIALIZABLE 
这是最高的隔离级别,它可以防此脏读,不可重复读和幻读等问题,但因其侵占式的数据记录完全锁定,导致它影响事务的性能,成为隔离级别中最展慢的一个。  
  
注意:并不是所有的资源管理器都支持所有的隔离级别,可针对不同的资源管理使用以上的隔离级别。


4.3.spring中的事务的只读属性:
在对数据库的操作中,查询是使用最频繁的操作,每次执行查询时都要从数据库中重新读取数据,有时多次读取的数据都是相同的,这样的数据操作不仅浪费了系统资源,还影响了系统速度。对访问量大的程序来说,节省这部分资源可以大大提升系统速度。 
如果将事务声明为只读的,那么数据库可以根据事务的特性优化事务的读取操作。事务的只读属性需要配合事务的传播行为共同设置。(如上例)


4.4.spring中的事务的超时属性:
这个属性和事务的只读属性一样需要搭配事务的传播行为共同设置,它设置了事务的超时时间,事务本身可能会因某种原因很长没有回应,在这期间事务可能锁定了数据库的表格,这样会出现严重的性能问题。通过设置事务的超时时间,从开始执行事务起,在规定的超时时间内如果没有事务就将它回滚。


5.spring的事务管理器:
spring的事务管理器有5个,都实现了PlatformTransactionManager接口,如下所示: 


DataSourceTransactionManager           JDBC事务管理器 
HibernateTransactionManager            Hibernate事务管理器 
JdoTransactionManager                  JDO事务管理器 
JtaTransactionManager                  JTA事务管理器      
PersistenceBrokerTransactionManager    Apache的OJB事务管理器      


6.spring的声明式事务:
因为Spring中的事务是基于AOP实现的,而Spring的AOP是以方法为单位的,所以Spring的事务属性就对事务应用到方法上的策略描述,这些属性为:传播行为,隔离级别,只读和超时属性。 
Spring的声明式事务不涉及组建依赖关系,它通过AOP实现事务管理。Spring本身就是一个容器,在使用Spring的声明式事务时不须编写任何代码,便可通过实现基于容器的事务管理。Spring提供了一些可供选择的辅助类,这些辅助类简化了传统的数据库操作流程,在一定程度上节省了工作量,提高了编码效率,所以推荐使用声明事务。


其优点如下:
I,优化DataSource:
DriverManagerDataSource数据源,它在每次获得数据连接时都创建一个新的连接对象,完全没有缓冲能力。 
Spring没有提供连接的实现,因为一些开源项目已经实现了带有连接池功能的数据源,例如Apache Commons DBCP的BasicDataSource数据源。这个数据源不仅提供了缓冲的连接池功能,而且它是一个完全轻量级的数据源。    


II,使用事务代理工厂: 
事务代理工厂TransactionProxyFactoryBean包括了事务拦截器,目标代理和事务的属性设置,它配置方便,使用灵活,在声明式事务中广泛使用。 
使用TransactionProxyFactoryBean需要注入它所依赖的事务管理器,设置代理的目标对象,代理对象的生成方式和事务属性。代理对象是在目标对象上生成的包括事物和AOP切面的新对象,这个新的对象用来替代目标对象以支持事物或AOP提供的切面功能。 


其中对象代理的生成方式可根据CGLIB生成代理,例如: 
<property name="proxyTargetClass" value="true"/> 


最后再介绍一个TransactionProxyFactoryBean中如何设置事务的属性。TransactionProxyFactoryBean的transactionAttributes属性用于指定事务的属性,它是一个Properties类型的属性集合,以方法名和事务属性组成键值对,给不同的方法指定不同的事务属性。


例如: <prop key="query*">PROPAGATION_REQUIRED,timeout_5,readOnly</prop> 


方法名作为声明时,可以使用*通配符,上述代码定义对所有以query作前缀的方法,并在定义的事务中完成操作,事务的不同属性之间以","分隔。 
通过一个实例介绍如何使用TransactionProxyFactoryBean完全成Spring的声明式事务管理。
0 0
原创粉丝点击