spring的事务管理
《一》使用spring提供的声明式事务管理功能(基于注解):
示例程序:
@Transactional //加上该注解,表明该类中的方法调用时默认都开启了事务进行管理
public class PersonServiceBean implements PersonService{
private JdbcTemplate jdbcTemplate;
public void setDataSource(DataSource dataSource){
this.jdbcTemplate = new JdbcTemplate(dataSource);
}
public void delete(Integer personid)throws Exception{
jdbcTemplate.update("delete from person where id=?",new Object[]{personid},new int[]{java.sql.Types.INTEGER});
throw new Exception("checked exception"); //虽然抛出了异常,但不会导致事务回滚,仍然会删去数据库中的一条记录。
}
。。。。。//其他的处理方法
}
总结:spring开启的事务,默认情况下,如果碰到unchecked exception,比如运行时异常(RuntimeExcetion)会导致事务回滚。
如果碰到checked exception,则不会导致事务回滚。但是我们可以利用注解属性,修改这些默认行为:
如:指定回滚
@Transactional (rollbackFor=Exception.class ) //指定即使出现checked exception,也会对事务进行回滚
public void delete(Integer personid)throws Exception{
jdbcTemplate.update("delete from person where id=?",new Object[]{personid},new int[]{java.sql.Types.INTEGER});
throw new Exception("checked exception");//抛出了异常,不会删除记录
}
指定不回滚
@Transactional (noRollbackFor=Exception.class ) //指定即使出现unchecked exception,也不会对事务进行回滚
public void delete(Integer personid)throws Exception{
jdbcTemplate.update("delete from person where id=?",new Object[]{personid},new int[]{java.sql.Types.INTEGER});
throw new RuntimeException("unchecked exception"); //抛出了异常,但会删除记录
}
有些方法不需要在事务中执行,也可以指定,比如从数据库中读取记录的行为:
@Transactional(propagation =Propagation.NOT_SUPPORTED )
public Person getPerson(Integer personid){
return (Person)jdbcTemplate.queryForObject("select * from person where id=?",new Object[]{personid},new int[]{java.sql.Types.INTEGER},new PersonRowMapper());
}
事务传播属性:
REQUIRED:业务方法需要在一个事务中运行。如果方法运行时,已经处在一个事务中,那么加入到该事务,否则为自己创建一个新的事务。
NOT_SUPPORTED:声明方法不需要事务。如果方法没有关联到一个事务,容器不会为它开启事务。如果方法在一个事务中被调用,该事务会被挂起,在方法调用结束后,原先的事务便会恢复执行。
REQUIRESNEW:属性表明不管是否存在事务,业务方法总会为自己发起一个新的事务。如果方法已经运行在一个事务中,则原有事务会被挂起,新的事务会被创建,知道方法执行结束,新事务才算结束,原有的事务才会恢复执行。
MANDATORY:该属性指定业务方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务。如果业务方法在没有事务的环境下调用,容器就会抛出异常。
SUPPORTS:这一事务属性表明,如果业务方法在某个事务范围内被调用,则方法成为该事务的一部分。如果业务方法在事务范围外被调用,则方法在没有事务的环境下执行。
NEVER:指定业务方法绝对不能在事务范围内执行。如果业务方法在某个事务中执行,容器胡抛出异常,只有业务方法没有关联到任何事务,才能正常执行。
NESTED:如果一个活动的事务存在,则运行在一个嵌套事务中。如果没有活动事务,则按REQUIRED属性执行。它使用了一个单独的事务 ,这个事务拥有多个可以回滚的保存点。内部事务的回滚不会对外部事务造成影响。 它只对DataSourceTransactionManager事务管理器起效。
nested详细分析:
@Resource OtherService otherService;
public void xxx(){
stmt.executeUpdate("update person set name='888' where id=1");
otherService.update();//OtherService的update方法的事务传播属性为 NESTED
stmt.executeUpdate("delete from person where id=9");
}
以上方法的内部执行过程如下:
Connection conn = null;
try{
conn.setAutoCommit(false);
Statement stmt = conn.createStatement();
Savepoint savepoint = conn.setSavepoint();
try{
conn.createStatement().executeUpdate("update person set name='222'where di=2"); //如果出现异常,不会导致外部事务的回滚,只会回滚到保存点
}catch(Exception ex){
conn.rollback(savepoint);
}
stmt.executeUpdate("delete from person where id=9"); //如果出现异常,会导致整个事务的回滚
conn.commit();
stmt.close();
}catch(Exception e){
conn.rollback();
}finally{
try{
if(null!=conn && !conn.isClosed()) conn.close();
}catch(SQLException e){e.printStackTrace();}
}