• Spring事务传递


    2018-09-25

        @Transactional(propagation=Propagation.NEVER)
        public void update(){
             Session s = sessionFactory.getCurrentSession();
             Student stu = (Student)s.get(Student.class, 7);
             System.out.println("ok" + stu.getName());
        }
        
        @Transactional(propagation=Propagation.REQUIRED)
        public void hasTran(){
            update();
        }

    hasTran和update方法如果在一个bean方法里面,调用hasTran()的时候,并不报错。说明update并没有识别出它是never级别的事务传递,因为在同一个bean里面调用不识别

    如果把hasTran()放到另一个bean里面,调用hasTran()就会报错。

    Existing transaction found for transaction marked with propagation 'never'

    如果两个id重复了,有保护机制。

    @Service
    public class TestService2 {

      @Transactional(propagation=Propagation.REQUIRED)
      public void hasTran(){
        Session s = sessionFactory.getCurrentSession();
        Student stu = new Student();
        stu.setName("spring2018+++");
        s.update(stu);
        testService.update();
      }

    }

    @Service
    public class TestService {

      @Transactional(propagation=Propagation.REQUIRED)
      public void update(){
        Session s = sessionFactory.getCurrentSession();
        Student stu = new Student();
        stu.setId(7);
        stu.setName("spring2016");
        s.update(stu);
      }

    }

    public static void main(String[] args) {

      TestService2 testService2 = (TestService2) context.getBean("testService2");
      testService2.hasTran();

    }

    Exception in thread "main" org.hibernate.NonUniqueObjectException: A different object with the same identifier value was already associated with the session : [cn.angelshelter.spring_study.pojo.Student#7]

    还有在一个事务内,如果多次调用Student stu = (Student)s.get(Student.class, 7);方法,其实只是查询了一次,不会说执行一次查询一次。

    如果方法A是REQUIRED的,方法B也是REQUIRED的,方法A调用方法B,方法B会抛出运行时异常,但是在A方法中,会try{ B },最后的结果是

    Exception in thread "main" org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only

    事务没有提交。

    如果在上面的基础上,把B改为REQUIRES_NEW,然后A和B不是同一个事务,A的事务是可以提交的。

    SUPPORTS最佛性,有事务就支持,没事务也通过。好像加了跟没加一个样。

    NOT_SUPPORTED。无事务运行,如果你遇到一个方法

     Exception in thread "main" org.hibernate.HibernateException: Could not obtain transaction-synchronized Session for current thread 
    那你加一个NOT_SUPPORTED它就不报错了。告诉程序,这个是没有事务也让它执行。
     
    MANDATORY 要求有事务,要不然抛出异常。
     
    刚发现REQUIRES_NEW有一定的危险性。有Studnet对象(id为7,张三),如果我在A方法中,把它的名字改为aa,然后调用B方法,B方法是REQUIRES_NEW的,这时候,如果把张三查询(按id查)出来,它的名字依然是张三,
    而不是方法中改变的aa。
     
    最后来一个总结吧:场景是A调用B,B抛出异常,A里面会捕获B的异常。
    REQUIRED  +  REQUIRED   他们是同一个事务,所以对象bean是互通的(就是A修改了Student, B如果查询,是查到A修改后的Student),然后B抛异常影响了A,A也提交不了。
     
    REQUIRED  +  NESTED      他们是同一个事务,所以对象bean是互通的,B抛异常不影响A(前提是要try), A最后可以提交事务
     
    REQUIRED  + REQUIRES_NEW  他们是不同事务,所以对象bean不是互通的,B抛异常不影响A(前提是要try), A最后可以提交事务
     
     
     
  • 相关阅读:
    Memento模式
    CSS实现半透明div层的方法
    JS解析json数据(如何将json字符串转化为数组)
    并发容器Map之二:ConcurrentNavigableMap
    JSinArray检查数组中是否存在某个值
    Servlet3.0之七:@WebFilter申明该类为过滤器
    Spring源码阅览——BeanFactory体系结构
    使用 Selenium RC 测试 web 应用程序
    函数式编程
    9 个 Java 处理 Exception 的最佳实践
  • 原文地址:https://www.cnblogs.com/angelshelter/p/9699344.html
Copyright © 2020-2023  润新知