首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 软件管理 > 软件架构设计 >

宣言式事务管理

2012-08-27 
声明式事务管理我们一下说的都是spring 整合hibernate的事物配置。Spring2.0及以后的版本中声明式事务有两

声明式事务管理
我们一下说的都是spring 整合hibernate的事物配置。
Spring2.0及以后的版本中声明式事务有两种风格的配置:使用基于XML文件和基于注解式的声明式事务配置方法。具体参见<Spring Framework 开发参考手册> 9.5 声明式事务管理。
Spring2.0及以后的版本中声明式事务的配置与之前的版本有相当大的不同。主要差异在于不再需要配置TransactionProxyFactoryBean了。
Spring2.0之前的旧版本风格的配置仍然是有效的;你可以简单地认为新的<tx:tags/>替你定义了TransactionProxyFactoryBean。
Spring2.0之前的旧版本风格的配置示例如下:

//applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd">

<beans default-dependency-check="none" default-autowire="no" default-lazy-init="false">

<bean id="dataSource" />
  </property>

  <property name="hibernateProperties">
   <props>
    <prop key="hibernate.dialect">org.hibernate.dialect.DB2Dialect</prop>
    <prop key="hibernate.show_sql">true</prop>
   </props>
  </property>
  <property name="mappingDirectoryLocations">
     <list>
    <value>classpath*:gov/cnca/aproduct/bo</value>
   <!--
     <value>classpath*:persist/role</value>
     <value>classpath*:persist/activity</value>
     <value>classpath*:persist/extension</value>
     <value>classpath*:persist/user</value>
           -->
           </list>
      
  </property>

  <!--
  <property name="mappingResources">
   <list>
    <value>gov/cnca/aproduct/bo/User.hbm.xml</value>
   </list>
  </property>
     -->
</bean>


<bean id="transactionManager" />
  </property>
</bean>

    <bean id="transactionProxy" abstract="true" parent="transactionProxy">
  <property name="target">
   <bean ref="iUserDAO" />
    <property name="userFriendsDAO" ref="iUserFriendsDAO" />
    <property name="userIntegralDAO" ref="iUserIntegralDAO" />
    <property name="userAwardDAO" ref="iUserAwardDAO" />
    <property name="luckRateDAO" ref="iLuckRateDAO" />
    <property name="rateDAO" ref="iRateDAO" />
   </bean>
  </property>
</bean>

<bean id="iSendEmailForPasswordService" parent="transactionProxy">
  <property name="target">
   <bean parent="transactionProxy">
  <property name="target">
   <bean ref="iUserDAO" />
    <property name="memberDAO" ref="iMemberDAO" />
    <property name="userIntegralDAO" ref="iUserIntegralDAO" />
    <property name="userAwardDAO" ref="iUserAwardDAO" />
   </bean>
  </property>
</bean>
<!-- dao -->
<bean id="iLuckRateDAO" name="iLuckRateDAO" ref="sessionFactory"></property>
    </bean>
<bean id="iRateDAO" name="iRateDAO" ref="sessionFactory"></property>
    </bean>
    <bean id="iUserDAO" name="iUserDAO" ref="sessionFactory"></property>
    </bean>
<bean id="iUserFriendsDAO" name="iUserFriendsDAO" ref="sessionFactory"></property>
    </bean>
<bean id="iUserIntegralDAO" name="iUserIntegralDAO" ref="sessionFactory"></property>
    </bean>
<bean id="iUserAwardDAO" name="iUserAwardDAO" ref="sessionFactory"></property>
    </bean>
<bean id="iMemberDAO" name="iMemberDAO" ref="sessionFactory"></property>
    </bean>

对于特定的方法或方法命名模式,代理的具体事务行为由事务属性驱动,如下面的例子所示:
<prop key="insert*">
ROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED
</prop>
key属性确定代理应该给哪个方法增加事务行为。这样的属性最重要的部份是传播行为。有以下选项可供使用:
PROPAGATION_REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
PROPAGATION_SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。
PROPAGATION_MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。
PROPAGATION_REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。
PROPAGATION_NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
PROPAGATION_NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。
PROPAGATION_NESTED--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作。
前六个策略类似于EJB CMT,第七个(PROPAGATION_NESTED)是Spring所提供的一个特殊变量。
它要求事务管理器或者使用JDBC 3.0 Savepoint API提供嵌套事务行为(如Spring的DataSourceTransactionManager)。
事务属性中的readOnly标志表示对应的事务应该被最优化为只读事务。这是一个最优化提示。在一些情况下,一些事务策略能够起到显著的最优化效果,
例如在使用Object/Relational映射工具(如:Hibernate或TopLink)时避免dirty checking(试图“刷新”)。
在事务属性中还有定义“timeout”值的选项,指定事务超时为几秒。在JTA中,这将被简单地传递到J2EE服务器的事务协调程序,并据此得到相应的解释。
------------------------------
Spring的事务管理是通过AOP代理实现的。其中的事务通知由元数据(目前基于XML或注解)驱动。代理对象与事务元数据结合产生了一个AOP代理,
它使用一个PlatformTransactionManager的实现配合TransactionInterceptor,在方法调用前后实施事务。spring使用aop机制管理jdbc的连接和事务。
它使用TransactionInterceptor类,Spring事务支持中的核心接口是org.springframework.transaction.PlatformTransactionManager。
为了实际执行事务,Spring所有的事务划分功能都通过传递适当的TransactionDefinition实例,委托给 PlatformTransactionManager。
尽管PlatformTransactionManager接口可以直接使用,应用程序通常配置具体的事务管理器并使用声明性事务来划分事务。Spring具有多种
PlatformTransactionManager实现,它们分为两类:
局部事务策略即针对单个资源执行事务(主要是针对单个的数据库)。实现有 org.springframework.jdbc.datasource.DataSourceTransactionManager。
它用于jdbc数据源的配置,调用TransactionInterceptor开启一个事务,从DataSource得到一个connection并确保auto-commit设为false。
他用JdbcTemplate在一个线程内绑定一个JDBC connection,TransactionInterceptor负责提交事务,DataSourceTransactionManager
调用Connection.commit()关闭connection,并解除绑定(potentially allowing for one thread connection per data source)。
例如:
<beans>
<bean id="DataSource" ref="DataSource" />
</bean>

<bean id="tatanTransactionScriptsProxyCreator" />
   </list>
  </property>
  <property name="interceptorNames">
   <list>
    <idref bean="DataSourceTransactionInterceptor" />
   </list>
  </property>
</bean>

<bean id="DataSourceTransactionInterceptor" ref="DataSourceTransactionManager" />
  <property name="transactionAttributeSource">
   <value>com.tatan.tatanTransactionScriptsImpl.*=PROPAGATION_REQUIRED</value>
  </property>
</bean>
</beans>

transactionAttributesSource 属性指定每个方法的transaction attribute,PROPAGATION_REQUIRED说明在一个事务内这个方法被执行。
和EJB一样,默认的情况下,spring只有当unchecked exception被抛出时,才rollback事务,也可以自己加入checked exception。
tatanTransactionScripts被TransactionInterceptor封装,在一个事物内执行类的每一个方法。

   全局事务管理即执行有可能跨越多个资源的全局事务。主要对应的Spring类是org.springframework.transaction.jta.JtaTransactionManager,它委托给遵循JTA规范的J2EE服务器,也有例外。
spring支持JTA,只需要一个标准的JtaTransactionManager定义,数据库必须支持XA protocol,或者J2EE服务器提供支持XA规范的DataSource。
默认的Spring JtaTransactionManager设置将从标准的JNDI位置获取JTA的 javax.transaction.UserTransaction对象,该JNDI位置由J2EE指定:java: comp/UserTransaction。对于大多数标准J2EE环境下的用例来说,它工作良好。
但是,默认的 JtaTransactionManager不能执行事务挂起操作(即它不支持PROPAGATION_REQUIRES_NEW和 PROPAGATION_NOT_SUPPORTED)。原因是标准的JTA UserTransaction接口不支持挂起或恢复事务的操作;它只支持开始和完成新事务的操作。
为执行事务挂起操作,还需要提供javax.transaction.TransactionManager实例,按照JTA的规定,它提供标准的挂起和恢复方法。遗憾的是,J2EE没有为JTA TransactionManager定义标准的JNDI位置!
因此,必须使用特定于供应商的(vendor-specific)查寻机制。J2EE没有考虑把JTA TransactionManager接口作为它的公开API的一部分。JTA规范规定的TransactionManager接口原本是打算用于容器集成的。
但是为JTA TransactionManager定义标准的JNDI位置还是有重大意义的,尤其是对于轻量级容器(如Spring);然后,便可以以同样的方式来定位任意的J2EE服务器的JTA TransactionManager。
结合jboss JTA的Spring事务划分
oracle-ds.xml
<?xml version="1.0" encoding="UTF-8"?>

<datasources>
<xa-datasource>
  <jndi-name>XASpringDS</jndi-name>
  <track-connection-by-tx/>
  <isSameRM-override-value>false</isSameRM-override-value>
  <xa-datasource-class>oracle.jdbc.xa.client.OracleXADataSource</xa-datasource-class>
  <xa-datasource-property name="URL">jdbc:oracle:oci8:@orcl</xa-datasource-property>
  <xa-datasource-property name="User">SCOTT</xa-datasource-property>
  <xa-datasource-property name="Password">tiger</xa-datasource-property>
  <exception-sorter-class-name>org.jboss.resource.adapter.jdbc.vendor.OracleExceptionSorter</exception-sorter-class-name>
  <no-tx-separate-pools/>
</xa-datasource>
<mbean
  code="org.jboss.resource.adapter.jdbc.xa.oracle.OracleXAExceptionFormatter"
  name="jboss.jca:service=OracleXAExceptionFormatter">
  <depends optional-attribute-name="TransactionManagerService">
   jboss:service=TransactionManager</depends>
</mbean>
</datasources>
spring配置
<!-- Data source bean -->

<bean id="dataSource" destroy-method="close">
  <property name="driverClassName" value="${jdbc.driverClassName}" />
  <property name="url" value="${jdbc.url}" />
  <property name="username" value="${jdbc.username}" />
  <property name="password" value="${jdbc.password}" />
</bean>
PlatformTransactionManager bean的定义如下:

<bean id="txManager" ref="dataSource"/>
</bean>
在dao中如下示例:
public class BigAmountRuleHibernateDAO extends BaseDaoHibernate implements BigAmountRuleDAO{

private JdbcTemplate jdbcTemplate;

  /**
   * jdbcTemplate根查询列表数据
   */
public List findListByJdbcTemplate(String sql,String[] args){
  List list = null;
  if(args != null)
   list=jdbcTemplate.queryForList(sql, args);
  else
   list=jdbcTemplate.queryForList(sql);
  return list;
}

public JdbcTemplate getJdbcTemplate() {
  return jdbcTemplate;
}

public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
  this.jdbcTemplate = jdbcTemplate;
}
}
(2)如果我们在J2EE容器里使用JTA,就像示例中 'dataAccessContext-jta.xml' 文件所示,我们将通过JNDI和Spring的 JtaTransactionManager 来获取一个容
器管理的 DataSource。JtaTransactionManager 不需要知道 DataSource 和其他特定的资源,因为它将使用容器提供的全局事务管理。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:jee="http://www.springframework.org/schema/jee"
    xsi:schemaLocation="
         http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
         http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.0.xsd">

  <jee:jndi-lookup id="dataSource" jndi-name="jdbc/jpetstore"/>

  <bean id="txManager" />

  <!-- other <bean/> definitions here -->

</beans>
注意
上面 'dataSource' 的bean定义使用了 'jee' 名称空间下的 <jndi-lookup/> 标签。想了解更多的配置信息, 请看附录 A, XML Schema-based configuration,
关于 <jee/> 标签的信息,可参考 第 A.2.3 节 “The jee schema” 节。

(3)我们也可以很容易地使用Hibernate局部事务,就像下面的Spring框架的 PetClinic 示例应用中的例子一样)。这种情况下,我们需要定义一个Hibernate的
LocalSessionFactoryBean,应用程序从中获取到Hibernate Session 实例。DataSource 的bean定义同上例类似(这里不再展示)。不过,
如果是一个JEE容器提供的 DataSource,它将由JEE容器自身,而不是Spring框架来管理事务。这种情况中'txManager' bean的类型为 HibernateTransactionManager。
同样地,DataSourceTransactionManager 需要一个指向 DataSource 的引用,而 HibernateTransactionManager 需要一个指向 SessionFactory 的引用。

<bean id="sessionFactory" ref="dataSource" />
  <property name="mappingResources">
    <list>
      <value>org/springframework/samples/petclinic/hibernate/petclinic.hbm.xml</value>
    </list>
  </property>
  <property name="hibernateProperties">
    <value>
   hibernate.dialect=${hibernate.dialect}
</value>
  </property>
</bean>

<bean id="txManager" ref="sessionFactory" />
</bean>

使用getHibernateTemplate()示例:
public class BigAmountRuleHibernate extends BaseDaoHibernate implements BigAmountRuleDAO{

public void saveOrUpdateEntity(Object obj){
  this.getHibernateTemplate().save(obj);
}

}
我们可以简单地使用 JtaTransactionManager 来处理Hibernate事务和JTA事务,就像我们处理JDBC,或者任何其它的资源策略一样。

<bean id="txManager" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:jee="http://www.springframework.org/schema/jee"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd
           http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.0.xsd">
<!--
  <jee:jndi-lookup id="dataSource" jndi-name="jdbc/MIBS48"/>
-->

<bean id="propertyConfigurer" value="/WEB-INF/jdbc.properties" />
  <!--
   <property name="locations"> <list>
   <value>/WEB-INF/jdbc.properties</value>
   <value>/WEB-INF/hiawardwebservice.properties</value> </list>
   </property>
  -->
</bean>
 
<!-- 支持 @Transactional 标记 --> 
<tx:annotation-driven proxy-target-transaction-manager="txManager"/>
 
<bean id="defaultLobHandler" lazy-init="false" />

<!-- 配置数据源
  <bean id="dataSource"
   value="${jdbc.driverClassName}" />
  <property name="url" value="${jdbc.url}" />
  <property name="username" value="${jdbc.username}" />
  <property name="password" value="${jdbc.password}" />
</bean>
 
<!--org.springframework.jdbc.datasource.DataSourceTransactionManager 或 org.springframework.orm.hibernate3.HibernateTransactionManager -->
<bean id="txManager" ref="dataSource" /> -- > <!-- 用DataSourceTransactionManager则需要此属性,而HibernateTransactionManager不需要。 -->
  <property name="sessionFactory">  <!-- 如果用DataSourceTransactionManager则去掉 -->
   <ref local="sessionFactory" />
  </property>
 
</bean>

<bean id="jdbcTemplate" />
  </property>
</bean>
<bean id="dataSourceJbpm" value="${jdbc.driverClassName}" />
  <property name="url" value="${jdbc.url}" />
  <property name="username" value="${jdbc.jbpmuser}" />
  <property name="password" value="${jdbc.password}" />
</bean>
<bean id="sessionFactoryJbpm" />
  </property>
  <property name="mappingJarLocations">
   <list>
   <value>WEB-INF/lib/jbpm-jpdl.jar</value>
   <value>WEB-INF/lib/jbpm-identity.jar</value>
   </list>
  </property>
  <property name="hibernateProperties">
   <props>
    <prop key="hibernate.show_sql">true</prop>
    <prop key="hibernate.format_sql">true</prop>
    <prop key="hibernate.dialect">
     org.hibernate.dialect.DB2Dialect
    </prop>
    <prop key="hibernate.cache.use_query_cache">true</prop>
    <!--
     <prop key="hibernate.cache.provider_class">
     org.hibernate.cache.EhCacheProvider </prop>
    -->
    <prop key="hibernate.cache.use_second_level_cache">
     true
    </prop>
    <prop key="hibernate.cache.provider_class">
     org.hibernate.cache.EhCacheProvider
    </prop>
    <prop key="hibernate.jdbc.batch_size">25</prop>
   </props>
  </property>
</bean>
<bean id="sessionFactory" ref="dataSource" />
 
  <property name="mappingDirectoryLocations">
   <list>
    <value>
     classpath:/com/hiaward/xbankmibs/cpc/dao/model
    </value>
    <value>
     classpath:/com/hiaward/xbankmibs/dao/journal/model
    </value>
    <value>
     classpath:/com/hiaward/xbankmibs/dao/taskmanager/model
    </value>
    <value>
     classpath:/com/hiaward/xbankmibs/dao/service/model
    </value>
    <value>
     classpath:/com/hiaward/xbankmibs/dao/security/model
    </value>
    <value>
     classpath:/com/hiaward/xbankmibs/message/model
    </value>
    <!-- 日常管理 映射文件 liangxuewei 2009-02-19 -->
    <value>
     classpath:/com/hiaward/xbankmibs/dao/daily/model
    </value>
          <value>
            classpath:/com/hiaward/xbankmibs/dao/accountIgt/model
          </value>
    <value>
     classpath:/com/hiaward/xbankmibs/dao/cardmade/model
    </value>
    <value>
            classpath:/com/hiaward/xbankmibs/dao/scenarioservice/model
          </value>
          <!-- 支付结算清差处理 -->
          <value>
            classpath:/com/hiaward/xbankmibs/dao/pay/model
          </value>
   </list>
  </property>
  <property name="hibernateProperties">
   <props>
    <prop key="hibernate.show_sql">true</prop>
    <prop key="hibernate.format_sql">true</prop>
    <prop key="hibernate.dialect">
     org.hibernate.dialect.DB2Dialect
    </prop>
    <prop key="hibernate.cache.use_query_cache">true</prop>
    <!--
     <prop key="hibernate.cache.provider_class">
     org.hibernate.cache.EhCacheProvider </prop>
    -->
    <prop key="hibernate.cache.use_second_level_cache">
     true
    </prop>
    <prop key="hibernate.cache.provider_class">
     org.hibernate.cache.EhCacheProvider
    </prop>
    <prop key="hibernate.jdbc.batch_size">25</prop>
   </props>
  </property>
</bean>
<!-- lob专用sessionFactory  -->
<bean id="sessionFactoryForLob"
  ref="dataSource" />
  <property name="lobHandler" ref="defaultLobHandler"/>
  <property name="mappingDirectoryLocations">
   <list>
    <value>classpath:/com/hiaward/xbankmibs/dao/service/model
    </value>
    <value>classpath:/com/hiaward/xbankmibs/dao/security/model
    </value>
    <value>classpath:/com/hiaward/xbankmibs/message/model
    </value>
    <value>
     classpath:/com/hiaward/xbankmibs/dao/journal/model
    </value>
   </list>
  </property>
  <property name="hibernateProperties">
   <props>
    <prop key="hibernate.show_sql">true</prop>
    <prop key="hibernate.format_sql">true</prop>
    <prop key="hibernate.dialect">
     org.hibernate.dialect.DB2Dialect
    </prop>
      <prop key="hibernate.cache.use_query_cache">true</prop>
     <prop key="hibernate.cache.provider_class">
     org.hibernate.cache.EhCacheProvider </prop>
    <prop key="hibernate.cache.use_second_level_cache">
     false
    </prop>
    <prop key="hibernate.cache.provider_class">
     org.hibernate.cache.EhCacheProvider
    </prop>
    <prop key="hibernate.jdbc.batch_size">0</prop> 
   </props>
  </property>
</bean>
</beans>

// Service:BigAmountRuleServiceImpl.java

@SuppressWarnings("unchecked")
@Transactional
public class BigAmountRuleServiceImpl implements BigAmountRuleService{
protected final Log log = LogFactory.getLog(getClass());
private BigAmountRuleDAO bigAmountRuleDAO;

/**
  * 查询条数
  * @param scenarioCode
  * @param orgCode
  * @param accountType
  * @param currencyCode
  * @return
  */
@Transactional (readOnly=false, isolation = Isolation.READ_COMMITTED) 
public int getTransRegisterCount(String scenarioCode,String orgCode,String accountType,String currencyCode){
  List<String[]> objTempList = null;
  if(scenarioCode != null || orgCode != null || accountType != null || currencyCode != null){
   String[] scenarioCodeArr = scenarioCode != null?scenarioCode.split(","):null;
   String[] orgCodeArr = orgCode != null?orgCode.split(","):null;
   String[] accountArr = accountType!=null?accountType.split(","):null;
   String[] currencyArr = currencyCode!=null?currencyCode.split(","):null;
  
   objTempList = new ArrayList<String[]>();
  
   objTempList.add(0,scenarioCodeArr);
   objTempList.add(1,orgCodeArr);
   objTempList.add(2,accountArr);
   objTempList.add(3,currencyArr);
  }
   
  int iCount = bigAmountRuleDAO.getTotalCount(objTempList);  // 使用了jdbcTemplate
  return iCount;
}
/**
  *
  * @param scenarioCode
  * @param orgCode
  * @param accountType
  * @param currencyCode
  * @return
  */
@Transactional (readOnly=false, isolation = Isolation.READ_COMMITTED) 
public Map<String,Object> getQueryDataList(String scenarioCode,String orgCode,String accountType,String currencyCode,int pStart,int size){
  List<String[]> objTempList = null;
  if(scenarioCode != null || orgCode != null || accountType != null || currencyCode != null){
   String[] scenarioCodeArr = scenarioCode != null?scenarioCode.split(","):null;
   String[] orgCodeArr = orgCode != null?orgCode.split(","):null;
   String[] accountArr = accountType!=null?accountType.split(","):null;
   String[] currencyArr = currencyCode!=null?currencyCode.split(","):null;
   objTempList = new ArrayList<String[]>();
  
   objTempList.add(0,scenarioCodeArr);
   objTempList.add(1,orgCodeArr);
   objTempList.add(2,accountArr);
   objTempList.add(3,currencyArr);

  }
   
  Map<String,Object> map = bigAmountRuleDAO.queryByList(objTempList, pStart, size); /*使用了getHibernateTemplate().方法名()和 回调
                       return this.getHibernateTemplate().executeFind(new HibernateCallback() {
                        public Object doInHibernate(Session session) throws SQLException,
                          HibernateException {
                         List vehicleList = null;
                         Query q = session.createQuery(sql);
                         q.setFirstResult(currentPage);
                         q.setMaxResults(pageSize);
                         vehicleList = q.list();
                         return vehicleList;
                         }
                       });
                      */
  return map;
}

@Transactional (readOnly=false, isolation = Isolation.READ_COMMITTED) 
public int getRecordCountBySql(String sql){
  return bigAmountRuleDAO.getListCount(sql);
}


public BigAmountRuleDAO getBigAmountRuleDAO() {
  return bigAmountRuleDAO;
}

public void setBigAmountRuleDAO(BigAmountRuleDAO bigAmountRuleDAO) {
  this.bigAmountRuleDAO = bigAmountRuleDAO;
}
}

//DAO:BigAmountRuleHibernate.java

@SuppressWarnings("unchecked")
public class BigAmountRuleHibernate extends BaseDaoHibernate implements BigAmountRuleDAO{

private JdbcTemplate jdbcTemplate;

public JdbcTemplate getJdbcTemplate() {
  return jdbcTemplate;
}

public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
  this.jdbcTemplate = jdbcTemplate;
}

public void saveOrUpdateEntity(Object obj){
  this.getHibernateTemplate().save(obj);
}

public void deleteEntity(Object obj){
  this.getHibernateTemplate().delete(obj);
}
  /**
   * jdbcTemplate根查询列表数据
   */
public List findListByJdbcTemplate(String sql,String[] args){
  List list = null;
  if(args != null)
   list=jdbcTemplate.queryForList(sql, args);
  else
   list=jdbcTemplate.queryForList(sql);
  return list;
}

public List findList(String sql,String[] args){
  return jdbcTemplate.queryForList(sql, args);
}

public Object getEntityByCondition(){
  return null;
}
/**
  * 新增和修改映射和大额核实规则信息
  */
public String saveByList(List<String> list)throws Exception{
  int iRef = 0;
  int iAchk = 0;
 
  Session session = this.getHibernateTemplate().getSessionFactory().openSession();
  try{
  Transaction tx = session.beginTransaction();
 
  for(int i=0;i<list.size();i++){
   String strArr = list.get(i);
   String[] arr = strArr.split(",");
  
   //新增前查看当前币种的规则是否已经存在
   String blnExistRuleSql = "select ref.rscerefcode from Rsceref ref,Amountcheck chk where ref.ruletype='11' and ref.rulecode = chk.bmcrcode and chk.monetarycategory=? and ref.scenariocode=? and ref.orgcode=?";
   List<Long> listExistRuleSql = this.getHibernateTemplate().find(blnExistRuleSql,new Object[]{arr[3],arr[0],arr[1]});
   //新增到规则
   Amountcheck chkObj = new Amountcheck();
   chkObj.setAccountcategory(arr[2]);
   chkObj.setMonetarycategory(arr[3]);
   chkObj.setMonetary(Double.valueOf(arr[4]));
   chkObj.setOperator(arr[5]);
   chkObj.setUpdatedatetime(new Date());
   this.getHibernateTemplate().save(chkObj);
   iAchk++;
   //新增映射
   if(listExistRuleSql != null && listExistRuleSql.size() > 0){//更新映射引用的规则ID
    String updateSql = "update Rsceref ref set ref.rulecode = ? where ref.rscerefcode = ?";
    getHibernateTemplate().bulkUpdate(updateSql, new Object[]{chkObj.getBmcrcode(),listExistRuleSql.get(0)});   
   }
   else{//当前币种对应的规则映射表中没有 直接新增
    Rsceref refObj = new Rsceref();
    refObj.setScenariocode(arr[0]);
    refObj.setOrgcode(arr[1]);
    refObj.setRulecode(chkObj.getBmcrcode());
    refObj.setRuletype("11");
    this.getHibernateTemplate().save(refObj);
    iRef++;
   }
  }
  tx.commit();
  }catch(Exception e){
   throw new Exception(e);
  }finally{
   if(session != null)
    session.close();
  
  }
  return iRef+","+iAchk;
}
/**
  * 批量删除
  */
public int deleteByList(List<String> list){
  StringBuffer sql = new StringBuffer();
  sql.append("from Rsceref ref where");
  sql.append(" ruletype = '11' and (");
  for(int i=0;i<list.size();i++){
   String strArr = list.get(i);
   String[] arr = strArr.split(",");
   //删除映射表
  
   sql.append("(ref.scenariocode='").append(arr[0]);
   sql.append("' and ref.orgcode='").append(arr[1]).append("' and ref.rulecode in");
   sql.append("(select chk.bmcrcode from Amountcheck chk where chk.accountcategory='").append(arr[2]);
   sql.append("' and chk.monetarycategory='").append(arr[3]).append("')) or");
  }
  String str = sql.substring(0, sql.length()-2);
  str = str+")";
  int iCount = Integer.parseInt(this.getHibernateTemplate().find("select count(*) "+str).get(0).toString());
  this.getHibernateTemplate().bulkUpdate("delete "+str);
  return iCount;
}
/**
  * 根据多选删除
  * @param args
  */
public void deleteBySelect(String[] args){
  StringBuffer strId = new StringBuffer();
  strId.append("delete from Rsceref where ruletype = '11' and rscerefcode in");
  for(int i=0;i<args.length;i++){
   if(i==0){
    strId.append("(");
    strId.append(args[i]);
   }
   else{
    strId.append(",");
    strId.append(args[i]);
   }
  }
  strId.append(")");
 
  jdbcTemplate.execute(strId.toString());
}

/**
  * 查询记录
  */
public Map<String,Object> queryByList(List<String[]> list,int pStart,int size){
  Map<String,Object> map = new HashMap<String,Object>();
  List listR = new ArrayList();
  StringBuffer sql=new StringBuffer();
  sql.append("from Rsceref ref,Amountcheck chk  where ref.rulecode = chk.bmcrcode and ref.ruletype='11'");
 
  if(list != null && list.size()>0){

   String[] scenarioCodeArr = list.get(0);
   String[] orgCodeArr = list.get(1);
   String[] accountArr = list.get(2);
   String[] currencyArr = list.get(3);
  
   if(scenarioCodeArr!=null && orgCodeArr != null && accountArr != null && currencyArr != null){
    sql.append(" and (");
    for(int i=0;i<scenarioCodeArr.length;i++){
     for(int j=0;j<orgCodeArr.length;j++){
      for(int k=0;k<accountArr.length;k++){
       for(int m=0;m<currencyArr.length;m++){
        sql.append("(ref.scenariocode='");
        sql.append(scenarioCodeArr[i]);
        sql.append("'");        

        sql.append(" and ref.orgcode='");
        sql.append(orgCodeArr[j]);
        sql.append("'");
       
        sql.append(" and chk.accountcategory='");
        sql.append(accountArr[k]);
        sql.append("'");
       
        sql.append(" and chk.monetarycategory='");
        sql.append(currencyArr[m]);
        sql.append("') or ");
       }
      }
     }
    }
    sql = new StringBuffer(sql.substring(0, sql.lastIndexOf("or")));
    sql.append(")");
   }
   else if(scenarioCodeArr!=null && orgCodeArr != null && accountArr != null){
    sql.append(" and (");
   
    for(int i=0;i<scenarioCodeArr.length;i++){
     for(int j=0;j<orgCodeArr.length;j++){
      for(int k=0;k<accountArr.length;k++){
       sql.append("(ref.scenariocode='");
       sql.append(scenarioCodeArr[i]);
       sql.append("'");        

       sql.append(" and ref.orgcode='");
       sql.append(orgCodeArr[j]);
       sql.append("'");
      
       sql.append(" and chk.accountcategory='");
       sql.append(accountArr[k]);
       sql.append("') or ");
      }
     }
    }
   
    sql = new StringBuffer(sql.substring(0, sql.lastIndexOf("or")));
    sql.append(")");
   }
  }

  sql.append(" order by ref.scenariocode,ref.orgcode,chk.accountcategory,chk.monetarycategory");

  listR = findPage(sql.toString(),pStart,size);

  List<Amountcheck> listEntity = new ArrayList();
  Map<String,String> mapName = new HashMap<String,String>();
 
  List listName = this.getHibernateTemplate().find("select strscenarionum,strscenarioname from Scenario");
 
  for(int i=0;i<listName.size();i++){
   Object[] obj = (Object[])listName.get(i);
   mapName.put(obj[0].toString(), obj[1].toString());
  }
  for(int i=0;i<listR.size();i++){
   Object[] obj = (Object[])listR.get(i);
   Rsceref refObj = (Rsceref)obj[0];
   Amountcheck chk = (Amountcheck)obj[1];
  
   chk.setRscerefcode(refObj.getRscerefcode());
   System.out.println(refObj.getScenariocode());
   chk.setScenarioName(mapName.get(refObj.getScenariocode()));
   chk.setScenarioCode(refObj.getScenariocode());
   chk.setOrgCode(refObj.getOrgcode());
   chk.setStrMonetary(NumberUtil.getNormalStyle(chk.getMonetary(), 2));
   listEntity.add(chk);
  }
  map.put("listEntity", listEntity);
  map.put("bigAmounChecktSql", sql.toString());
  return map;
}
/**
  * 分页方法
  */
public List findPage(final String sql, final int currentPage,final int pageSize) {
  return this.getHibernateTemplate().executeFind(new HibernateCallback() {
   public Object doInHibernate(Session session) throws SQLException,
     HibernateException {
    List vehicleList = null;
    Query q = session.createQuery(sql);
    q.setFirstResult(currentPage);
    q.setMaxResults(pageSize);
    vehicleList = q.list();
    return vehicleList;
   }
  });
}
}


以action -> service -> dao 这种调用路径执行时,因为通过@Transactional我们把连接点(joinpoint)加入到service层,所以
在调用service的方法如getTransRegisterCount()还未进入时,就会调用TransactionInterceptor开启一个事务,进入
Cglib2AopProxy$CglibMethodInvocation(ReflectiveMethodInvocation).proceed(),如下
代码1.

public Object proceed() throws Throwable {
  // We start with an index of -1 and increment early.
  //interceptorsAndDynamicMethodMatchers中包含[org.springframework.transaction.interceptor.TransactionInterceptor@1b53927]
  if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
   return invokeJoinpoint();
  }

  Object interceptorOrInterceptionAdvice =
      this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);
  if (interceptorOrInterceptionAdvice instanceof InterceptorAndDynamicMethodMatcher) {
   // Evaluate dynamic method matcher here: static part will already have
   // been evaluated and found to match.
   InterceptorAndDynamicMethodMatcher dm =
       (InterceptorAndDynamicMethodMatcher) interceptorOrInterceptionAdvice;
   if (dm.methodMatcher.matches(this.method, this.targetClass, this.arguments)) {
    return dm.interceptor.invoke(this);
   }
   else {
    // Dynamic matching failed.
    // Skip this interceptor and invoke the next in the chain.
    return proceed();
   }
  }//与切入点匹配,调用匹配成功的方法,因为我们是对所有service的public方法开启事物,以这里为例就是
  else {
   // It's an interceptor, so we just invoke it: The pointcut will have
   // been evaluated statically before this object was constructed.
   // 调用interceptor:TransactionInterceptor.java 的invoke() ,见代码2.
   return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this);
  }
}

代码2.
// TransactionTnterceptor.java
public Object invoke(final MethodInvocation invocation) throws Throwable {
  // Work out the target class: may be <code>null</code>.
  // The TransactionAttributeSource should be passed the target class
  // as well as the method, which may be from an interface.
  // 这里对应的是你的service:BigAmountRuleServiceImpl
  Class targetClass = (invocation.getThis() != null ? invocation.getThis().getClass() : null);

  // If the transaction attribute is null, the method is non-transactional.
  final TransactionAttribute txAttr =
    getTransactionAttributeSource().getTransactionAttribute(invocation.getMethod(), targetClass);
  final String joinpointIdentification = methodIdentification(invocation.getMethod());

  if (txAttr == null || !(getTransactionManager() instanceof CallbackPreferringPlatformTransactionManager)) {
   // Standard transaction demarcation with getTransaction and commit/rollback calls.
   // TransactionInfo 中拥有TransactionAttribute,TransactionStatus引用,TransactionStatus(DefaultTransactionStatus)中拥有Object transaction。
   // 创建事物,txAttr包含隔离级别、传播行为、是否只读。joinpointIdentification = com.hiaward.xbankmibs.cpc.service.impl.BigAmountRuleServiceImpl.getRecordCountBySql
   // 见代码3.
   TransactionInfo txInfo = createTransactionIfNecessary(txAttr, joinpointIdentification);
   Object retVal = null;
   try {
    // This is an around advice: Invoke the next interceptor in the chain.
    // This will normally result in a target object being invoked.
    /*
    * 执行代码1.的
    *
    * if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
    *  return invokeJoinpoint();
    * }
    * 执行代码代码15.
    *
    * 之后调用service的方法,这里是getRecordCountBySql(String sql),在执行DAO的方法。
    *  public int getListCount(String sql){
    *
    *  return this.getHibernateTemplate().find(sql).size();
    *
    *  }
    * 见代码10.
    */
    retVal = invocation.proceed();
   }
   catch (Throwable ex) {
    // target invocation exception
    completeTransactionAfterThrowing(txInfo, ex);
    throw ex;
   }
   finally {
    cleanupTransactionInfo(txInfo); // Reset the TransactionInfo ThreadLocal.
   }
   // 刷新、提交、关闭connection。见代码16. 这样整个流程跑完了。
   commitTransactionAfterReturning(txInfo);
   return retVal;
  }

  else {
   // It's a CallbackPreferringPlatformTransactionManager: pass a TransactionCallback in.
   try {
    Object result = ((CallbackPreferringPlatformTransactionManager) getTransactionManager()).execute(txAttr,
      new TransactionCallback() {
       public Object doInTransaction(TransactionStatus status) {
        TransactionInfo txInfo = prepareTransactionInfo(txAttr, joinpointIdentification, status);
        try {
         return invocation.proceed();
        }
        catch (Throwable ex) {
         if (txAttr.rollbackOn(ex)) {
          // A RuntimeException: will lead to a rollback.
          if (ex instanceof RuntimeException) {
           throw (RuntimeException) ex;
          }
          else {
           throw new ThrowableHolderException(ex);
          }
         }
         else {
          // A normal return value: will lead to a commit.
          return new ThrowableHolder(ex);
         }
        }
        finally {
         cleanupTransactionInfo(txInfo);
        }
       }
      });

    // Check result: It might indicate a Throwable to rethrow.
    if (result instanceof ThrowableHolder) {
     throw ((ThrowableHolder) result).getThrowable();
    }
    else {
     return result;
    }
   }
   catch (ThrowableHolderException ex) {
    throw ex.getCause();
   }
  }
}

代码3. TransactionAspectSupport.java
/**
  * Create a transaction if necessary based on the given TransactionAttribute.
  * <p>Allows callers to perform custom TransactionAttribute lookups through
  * the TransactionAttributeSource.
  * @param txAttr the TransactionAttribute (may be <code>null</code>)
  * @param joinpointIdentification the fully qualified method name
  * (used for monitoring and logging purposes)
  * @return a TransactionInfo object, whether or not a transaction was created.
  * The <code>hasTransaction()</code> method on TransactionInfo can be used to
  * tell if there was a transaction created.
  * @see #getTransactionAttributeSource()
  */
protected TransactionInfo createTransactionIfNecessary(TransactionAttribute txAttr, final String joinpointIdentification) {

  // If no name specified, apply method identification as transaction name.
  // 如果事物没有指定名字,则把方法标识作为事物名。
  if (txAttr != null && txAttr.getName() == null) {
   txAttr = new DelegatingTransactionAttribute(txAttr) {
    public String getName() {
     return joinpointIdentification;
    }
   };
  }

  TransactionStatus status = null;
  if (txAttr != null) {
   // Return the transaction manager.这里是org.springframework.orm.hibernate3.HibernateTransactionManager
   PlatformTransactionManager tm = getTransactionManager();
   if (tm != null) {
    // 得到事物状态对象(newTransactionStatus(definition, null, true, newSynchronization, debugEnabled, null);))
    // 见代码4.
    status = tm.getTransaction(txAttr);
   }
   else {
    if (logger.isDebugEnabled()) {
     logger.debug("Skipping transactional joinpoint [" + joinpointIdentification +
       "] because no transaction manager has been configured");
    }
   }
  }
  return prepareTransactionInfo(txAttr, joinpointIdentification, status);
}

代码4.
/**
  * This implementation handles propagation behavior. Delegates to
  * <code>doGetTransaction</code>, <code>isExistingTransaction</code>
  * and <code>doBegin</code>.
  * @see #doGetTransaction
  * @see #isExistingTransaction
  * @see #doBegin
  */
public final TransactionStatus getTransaction(TransactionDefinition definition) throws TransactionException {
  // 返回DataSourceTransactionObject 对象txObject,并txObject.setConnectionHolder(conHolder, false),
  // conHolder可能是null(如果TransactionSynchronizationManager的ThreadLocal resources 中不存在这个ConnectionHolder)。
  // 见代码5.
  Object transaction = doGetTransaction();

  // Cache debug flag to avoid repeated checks.
  boolean debugEnabled = logger.isDebugEnabled();
  //interface TransactionAttribute extends TransactionDefinition
  if (definition == null) {
   // Use defaults if no transaction definition given.
   definition = new DefaultTransactionDefinition();
  }
 
  //transaction.getConnectionHolder() != null && transaction.getConnectionHolder().isTransactionActive()
  if (isExistingTransaction(transaction)) {
   // Existing transaction found -> check propagation behavior to find out how to behave.
   return handleExistingTransaction(definition, transaction, debugEnabled);
  }

  // Check definition settings for new transaction.
  if (definition.getTimeout() < TransactionDefinition.TIMEOUT_DEFAULT) {
   throw new InvalidTimeoutException("Invalid transaction timeout", definition.getTimeout());
  }

  // No existing transaction found -> check propagation behavior to find out how to proceed.
  if (definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_MANDATORY) {
   throw new IllegalTransactionStateException(
     "No existing transaction found for transaction marked with propagation 'mandatory'");
  }
  else if (definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRED ||
     definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRES_NEW ||
           definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_NESTED) {
          
   SuspendedResourcesHolder suspendedResources = suspend(null);
   if (debugEnabled) {
    logger.debug("Creating new transaction with name [" + definition.getName() + "]: " + definition);
   }
   try {
    // 设置事物(DataSourceTransactionObject)对象txObject的属性,如isolation level(隔离级别),
    // 绑定ConnectionHolder和SessionHolder到TransactionSynchronizationManager的ThreadLocal resources 中。
    // 见代码6.事物开启。
    doBegin(transaction, definition);
   }
   catch (RuntimeException ex) {
    resume(null, suspendedResources);
    throw ex;
   }
   catch (Error err) {
    resume(null, suspendedResources);
    throw err;
   }
   boolean newSynchronization = (getTransactionSynchronization() != SYNCHRONIZATION_NEVER);
  
   // Create a new TransactionStatus(DefaultTransactionStatus ,其中包含transaction属性) for the given arguments ,
   // initializing transaction synchronization as appropriate(TransactionSynchronizationManager的ThreadLocal 变量).
   return newTransactionStatus(definition, transaction, true, newSynchronization, debugEnabled, suspendedResources); 
  }
  else {
   // Create "empty" transaction: no actual transaction, but potentially synchronization.
   boolean newSynchronization = (getTransactionSynchronization() == SYNCHRONIZATION_ALWAYS);
   return newTransactionStatus(definition, null, true, newSynchronization, debugEnabled, null);
  }
}

代码5.
/**
* 返回一个事物对象HibernateTransactionObject,并对其sessionHolder、connectionHolder属性赋值。
*
*/
protected Object doGetTransaction() {
  HibernateTransactionObject txObject = new HibernateTransactionObject();
  txObject.setSavepointAllowed(isNestedTransactionAllowed());

     // 从(当前线程绑定的)变量resource中查找是否存在(用来装载Session的)SessionHolder,
     // 这个变量resource由TransactionSynchronizationManager来维护。如果存在则放到线程对象txObject中。
  SessionHolder sessionHolder = (SessionHolder) TransactionSynchronizationManager.getResource(getSessionFactory());
  if (sessionHolder != null) {
   if (logger.isDebugEnabled()) {
    logger.debug("Found thread-bound Session [" + SessionFactoryUtils.toString(sessionHolder.getSession()) + "] for Hibernate transaction");
   }
   txObject.setSessionHolder(sessionHolder);
  }
  else if (this.hibernateManagedSession) {
   try {
    Session session = getSessionFactory().getCurrentSession();
    if (logger.isDebugEnabled()) {
     logger.debug("Found Hibernate-managed Session [" + SessionFactoryUtils.toString(session) + "] for Spring-managed transaction");
    }
    txObject.setExistingSession(session);
   }
   catch (HibernateException ex) {
    throw new DataAccessResourceFailureException( "Could not obtain Hibernate-managed Session for Spring-managed transaction", ex);
   }
  }

  if (getDataSource() != null) {
 
   // 从(当前线程绑定的)变量resource中查找是否存在(用来装载Connection的)ConnectionHolder,
      // 这个变量resource由TransactionSynchronizationManager来维护。如果存在则放到线程对象txObject中。
   ConnectionHolder conHolder = (ConnectionHolder)TransactionSynchronizationManager.getResource(getDataSource());
   txObject.setConnectionHolder(conHolder);
  }

  return txObject;
}

代码6.
protected void doBegin(Object transaction, TransactionDefinition definition) {
  HibernateTransactionObject txObject = (HibernateTransactionObject) transaction;

  if (txObject.hasConnectionHolder() && !txObject.getConnectionHolder().isSynchronizedWithTransaction()) {
   throw new IllegalTransactionStateException(
     "Pre-bound JDBC Connection found! HibernateTransactionManager does not support " +
     "running within DataSourceTransactionManager if told to manage the DataSource itself. " +
     "It is recommended to use a single HibernateTransactionManager for all transactions " +
     "on a single DataSource, no matter whether Hibernate or JDBC access.");
  }

  Session session = null;

  try {
   if (txObject.getSessionHolder() == null || txObject.getSessionHolder().isSynchronizedWithTransaction()) {
    Interceptor entityInterceptor = getEntityInterceptor();
    // getSessionFactory().openSession() 见代码7.
    Session newSession = (entityInterceptor != null ?
      getSessionFactory().openSession(entityInterceptor) : getSessionFactory().openSession());
     
    if (logger.isDebugEnabled()) {
     logger.debug("Opened new Session [" + SessionFactoryUtils.toString(newSession) +
       "] for Hibernate transaction");
    }
    // 把事物对象session属性赋值。这里Session对象的connection = null
    txObject.setSession(newSession);
   }

   session = txObject.getSessionHolder().getSession();

   if (this.prepareConnection && isSameConnectionForEntireSession(session)) {
    // We're allowed to change the transaction settings of the JDBC Connection.
    if (logger.isDebugEnabled()) {
     logger.debug(
       "Preparing JDBC Connection of Hibernate Session [" + SessionFactoryUtils.toString(session) + "]");
    }
    // 从Session中建立连接,见代码8.
    Connection con = session.connection();
    // 根据条件是否执行con.setReadOnly(true),con.setTransactionIsolation(definition.getIsolationLevel())。然后返回previousIsolationLevel。
    Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);
    txObject.setPreviousIsolationLevel(previousIsolationLevel);
   }
   else {
    // Not allowed to change the transaction settings of the JDBC Connection.
    if (definition.getIsolationLevel() != TransactionDefinition.ISOLATION_DEFAULT) {
     // We should set a specific isolation level but are not allowed to...
     throw new InvalidIsolationLevelException(
       "HibernateTransactionManager is not allowed to support custom isolation levels: " +
       "make sure that its 'prepareConnection' flag is on (the default) and that the " +
       "Hibernate connection release mode is set to 'on_close' (SpringTransactionFactory's default). " +
       "Make sure that your LocalSessionFactoryBean actually uses SpringTransactionFactory: Your " +
       "Hibernate properties should *not* include a 'hibernate.transaction.factory_class' property!");
    }
    if (logger.isDebugEnabled()) {
     logger.debug(
       "Not preparing JDBC Connection of Hibernate Session [" + SessionFactoryUtils.toString(session) + "]");
    }
   }
  
   // 如果是definition.isReadOnly() = TRUE ,为查询,所以不用与DB同步。因为不涉及更新DB内容。
   if (definition.isReadOnly() && txObject.isNewSession()) {
    // Just set to NEVER in case of a new Session for this transaction.
    session.setFlushMode(FlushMode.NEVER);
   }

   // 如果是txObject.isNewSession() = TRUE ,则不用设置session.setFlushMode(FlushMode.AUTO),因为新Session 默认就是FlushMode.AUTO。
   if (!definition.isReadOnly() && !txObject.isNewSession()) {
    // We need AUTO or COMMIT for a non-read-only transaction.
    FlushMode flushMode = session.getFlushMode();
    if (flushMode.lessThan(FlushMode.COMMIT)) {
     session.setFlushMode(FlushMode.AUTO);
     txObject.getSessionHolder().setPreviousFlushMode(flushMode);
    }
   }

   Transaction hibTx = null;

   // Register transaction timeout.
   int timeout = determineTimeout(definition);
   if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
    // Use Hibernate's own transaction timeout mechanism on Hibernate 3.1
    // Applies to all statements, also to inserts, updates and deletes!
    hibTx = session.getTransaction();
    hibTx.setTimeout(timeout);
    hibTx.begin();
   }
   else {
    // Open a plain Hibernate transaction without specified timeout.
    //打开连接,设置con.setAutoCommit(false),开启事物,去掉自动提交。见代码9.
    hibTx = session.beginTransaction();
   }

   // Add the Hibernate transaction to the session holder.
   txObject.getSessionHolder().setTransaction(hibTx);

   // Register the Hibernate Session's JDBC Connection for the DataSource, if set.
   if (getDataSource() != null) {
    //这里通过session打开的连接是同一个连接
    Connection con = session.connection();
    ConnectionHolder conHolder = new ConnectionHolder(con);
    if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
     conHolder.setTimeoutInSeconds(timeout);
    }
    if (logger.isDebugEnabled()) {
     logger.debug("Exposing Hibernate transaction as JDBC transaction [" + con + "]");
    }
    // Bind the Connection holder to the thread.
    TransactionSynchronizationManager.bindResource(getDataSource(), conHolder);
    txObject.setConnectionHolder(conHolder);
   }

   // Bind the session holder to the thread.
   if (txObject.isNewSessionHolder()) {
    TransactionSynchronizationManager.bindResource(getSessionFactory(), txObject.getSessionHolder());
   }
   txObject.getSessionHolder().setSynchronizedWithTransaction(true);
  }

  catch (Exception ex) {
   if (txObject.isNewSession()) {
    try {
     if (session.getTransaction().isActive()) {
      session.getTransaction().rollback();
     }
    }
    catch (Throwable ex2) {
     logger.debug("Could not rollback Session after failed transaction begin", ex);
    }
    finally {
     SessionFactoryUtils.closeSession(session);
    }
   }
   throw new CannotCreateTransactionException("Could not open Hibernate Session for transaction", ex);
  }
}

代码7. SessionFactoryImpl

public org.hibernate.classic.Session openSession() throws HibernateException {
  return openSession(interceptor);
}

public org.hibernate.classic.Session openSession(Interceptor sessionLocalInterceptor) throws HibernateException {
  // note that this timestamp is not correct if the connection provider
  // returns an older JDBC connection that was associated with a
  // transaction that was already begun before openSession() was called
  // (don't know any possible solution to this!)
 
  long timestamp = settings.getRegionFactory().nextTimestamp();
  return openSession( null, true, timestamp, sessionLocalInterceptor );
}

private SessionImpl openSession(Connection connection,boolean autoClose,long timestamp,Interceptor sessionLocalInterceptor) {
  return new SessionImpl(
          connection,
          this,
          autoClose,
          timestamp,
          sessionLocalInterceptor == null ? interceptor : sessionLocalInterceptor,
          settings.getDefaultEntityMode(),
          settings.isFlushBeforeCompletionEnabled(),
          settings.isAutoCloseSessionEnabled(),
          settings.getConnectionReleaseMode()
   );
}
从这里看出由SessionFactory每次都是产生一个新的Session对象(SessionImpl)。

代码8. SessionImpl.java

public Connection connection() throws HibernateException {
  errorIfClosed();
  return jdbcContext.borrowConnection();
}

// JDBCContext.java
public Connection borrowConnection() {
  return connectionManager.borrowConnection();
}

// ConnectionManager.java
public Connection borrowConnection() {
  if ( isClosed ) {
   throw new HibernateException( "connection manager has been closed" );
  }
  if ( isSuppliedConnection() ) {
   return connection;
  }
  else {
   if ( borrowedConnection == null ) {
    borrowedConnection = BorrowedConnectionProxy.generateProxy( this );
   }
   return borrowedConnection;
  }
}
// BorrowedConnectionProxy.java
/**
  * Generates a Connection proxy wrapping the connection managed by the passed
  * connection manager.
  *
  * @param connectionManager The connection manager to wrap with the

热点排行