zoukankan      html  css  js  c++  java
  • JAVA框架之Spring【Spring事务详解】

    spring提供的事务管理可以分为两类:编程式的和声明式的。编程式的,比较灵活,但是代码量大,存在重复的代码比较多;声明式的比编程式的更灵活。编程式主要使用transactionTemplate。省略了部分的提交,回滚,一系列的事务对象定义,需注入事务管理对象;声明式:主要使用TransactionProxyFactoryBean,围绕Poxy的动态代理,能够自动的提交和回滚事务。统观spring事务,围绕着两个核心PlatformTransactionManager和TransactionStatus,下面我们以声明式方式来讲解。
    Spring声明式是以代理的方式实现对事务的管理。我们在Action中所使用的Service对象,其实是代理对象的实例,并不是我们所写的Service对象实例。既然是两个不同的对象,那为什么我们在Action中可以象使用Service对象一样的使用代理对象呢?为了说明问题,假设有个Service类叫AService,它的Spring事务代理类为AProxyService,AService实现了一个接口 IAService (这里多了一个接口IAService,是为了说明接口代理的方式)。

    Spring事务代理方式有两种,一种是类代理方式,一种是接口代理方式。在Spring的配置文件中可以指定代理方式,目前我们使用的都是类代理方式。 //以下配置指定为类代理方式

    <property name="proxyTargetClass"><value>true</value></property>

    1、类代理方式

    类代理方式的实现方式是通过继承的方式来实现,下面用伪代码来说明。

    interface IAService{
      public void save(ValueObject object);

    }

    class AService implements IAService{
      public void save(ValueObject object){ ... }

    }

    // Spring自动生成的代理类.

    class AProxyService extends AService{
      public void save(ValueObject object){
          try{
             启动事务的代码;
              super.save(object);
             提交事务的代码;
          }catch(Exception e){
             回滚事务的代码;
          }
      }

    }

    在Action的调用代码:

    AService a = (AService)getBean("aProxyServiceBeanName");

    这里 getBean("aProxyServiceBeanName") 得到的是 AProxyService 类的实例,因为AProxyService是AService的一个子类,所以这里可以强制转型为AService。我们后面调用a.save(object)的时候,调用的是AProxyService.save()方法,而这个方法是有事务处理的。Spring就是这样实现了事务管理。 // 下面的这行代码也是同样的效果。

    IAService a = (IAService)getBean("aProxyServiceBeanName");

    Spring通过CGLib来实现了类代理方式。


    2、接口代理方式

    接口代理方式是通过实现接口,引用类实例来实现的,所以这里一定要有一个接口IAService,而类代理方式是不需要这个接口的。

    // Spring自动生成的代理类.

    class AProxyService implements IAService{
      private AService aService;
       public void setAService(AService aService){
         this.aService = aService;
      }
       public void save(ValueObject object){
          try{
             启动事务的代码;
             aService.save(object); // 注意这行代码与上面的不同。
             提交事务的代码;
          }catch(Exception e){
             回滚事务的代码;
          }
      }

    }

    在Action的调用代码:

    AService a = (AService)getBean("aProxyServiceBeanName");

    上面这行代码会报ClassCastException错误,因为 getBean("aProxyServiceBeanName") 得到的是 AProxyService 类的实例,而AProxyService实例并不能转型为AService类型,虽然两者都实现了同一个接口,但他们之间并没有继承关系。就象ArrayList 和 LinkedList,他们可以转换成List,但他们之间并不能互相转换。所以这里必须这样使用:

    IAService a = (IAService)getBean("aProxyServiceBeanName"); // 要转换成接口类型。

    调用a.save() 方法其实就是调用 AProxyService.save() 方法。

    Spring通过Java动态代理来实现接口代理
  • 相关阅读:
    Linux系统-命令行快捷键
    Golang理解-Context包
    Golang理解-垃圾回收机制
    Linux系统-Systemd资源控制
    Linux系统-"cannot access Input/output error"
    Linux系统-ntpdate时间同步报错
    Golang理解-数组和切片
    Golang理解-位运算
    Golang理解-指针
    Maven配置,使用IntelliJ IDEA和Maven创建Java Web项目
  • 原文地址:https://www.cnblogs.com/hnlshzx/p/3497604.html
Copyright © 2011-2022 走看看