Spring 多数据源事务配置问题

在SpringSide 3 中,白衣提供的预先配置好的环境非常有利于用户进行快速开发,但是同时也会为扩展带来一些困难。最直接的例子就是关于在项目中使用多个数据源的问题,似乎 很难搞。在上一篇中,我探讨了SpringSide 3 中的数据访问层,在这一篇中,我立志要解决多数据源配置的难题,我的思路是这样的: 

第一步、测试能否配置多个DataSource 
第二步、测试能否配置多个SessionFactory 
第三步、测试能否配置多个TransactionManager 
第四步、测试能否使用多个TransactionManager,也就是看能否配置多个 

基本上到第四步就应该走不通了,因为Spring中似乎不能配置多个,而且@transactional注解也无法让用户选择具体使用哪个TransactionManager。也就是说,在SpringSide的应用中,不能让不同的数据源分别属于不同的事务管理器,多数据源只能使用分布式事务管理器,那么测试思路继续如下进行: 

第五步、测试能否配置JTATransactionManager 

如果到这一步,项目还能顺利在Tomcat中运行的话,我们就算大功告成了。但我总认为事情不会那么顺利,我总觉得JTATransactionManager需要应用服务器的支持,而且需要和JNDI配合使用,具体是不是这样,那只有等测试后才知道。如果被我不幸言中,那么进行下一步: 

第六步、更换Tomcat为GlassFish,更换JDBC的DataSource为JNDI查找的DataSource,然后配置JTATransactionManager 

下面测试开始,先假设场景,还是继续用上一篇中提到的简单的文章发布系统,假设该系统运行一段时间后非常火爆,单靠一台服务器已经无法支持巨大的用户数, 这时候,站长想到了把数据进行水平划分,于是,需要建立一个索引数据库,该索引数据库需保存每一篇文章的Subject及其内容所在的Web服务器,而每 一个Web服务器上运行的项目,需要同时访问索引数据库和内容数据库。所以,需要创建索引数据库,如下: 
Java代码   收藏代码
  1. create database puretext_index;  
  2.   
  3. use puretext_index;  
  4.   
  5. create table articles(  
  6. id int primary key auto_increment,  
  7. subject varchar(256),  
  8. webserver varchar(30)  
  9. );  

第一步测试,配置多个DataSource,配置文件如下: 
application.properties: 
Java代码   收藏代码
  1. jdbc.urlContent=jdbc:mysql://localhost:3306/PureText?useUnicode=true&characterEncoding=utf8  
  2. jdbc.urlIndex=jdbc:mysql://localhost:3306/PureText_Index?useUnicode=true&characterEncoding=utf8  

applicationContext.xml: 
Java代码   收藏代码
  1. "1.0" encoding="UTF-8"?>  
  2. "http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  3.     xmlns:jee="http://www.springframework.org/schema/jee" xmlns:tx="http://www.springframework.org/schema/tx"  
  4.     xmlns:context="http://www.springframework.org/schema/context"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"  
  6.     default-lazy-init="true">  
  7.   
  8.     Spring公共配置文件   
  9.   
  10.       
  11.     class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">  
  12.         "systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE" />  
  13.         "ignoreResourceNotFound" value="true" />  
  14.         "locations">  
  15.               
  16.                   
  17.                 classpath*:/application.properties  
  18.                   
  19.                 classpath*:/application.local.properties  
  20.                   
  21.                   
  22.               
  23.           
  24.       
  25.   
  26.       
  27.     package="cn.puretext" />  
  28.   
  29.       
  30.     "dataSourceContent" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">  
  31.           
  32.         "driverClassName" value="com.mysql.jdbc.Driver" />  
  33.         "url" value="${jdbc.urlContent}" />  
  34.         "username" value="${jdbc.username}" />  
  35.         "password" value="${jdbc.password}" />  
  36.   
  37.           
  38.         "initialSize" value="5" />  
  39.         "maxActive" value="100" />  
  40.         "maxIdle" value="30" />  
  41.         "maxWait" value="1000" />  
  42.         "poolPreparedStatements" value="true" />  
  43.         "defaultAutoCommit" value="false" />  
  44.       
  45.     "dataSourceIndex" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">  
  46.           
  47.         "driverClassName" value="com.mysql.jdbc.Driver" />  
  48.         "url" value="${jdbc.urlIndex}" />  
  49.         "username" value="${jdbc.username}" />  
  50.         "password" value="${jdbc.password}" />  
  51.   
  52.           
  53.         "initialSize" value="5" />  
  54.         "maxActive" value="100" />  
  55.         "maxIdle" value="30" />  
  56.         "maxWait" value="1000" />  
  57.         "poolPreparedStatements" value="true" />  
  58.         "defaultAutoCommit" value="false" />  
  59.       
  60.   
  61.       
  62.       
  63.   
  64.       
  65.     "sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">  
  66.         "dataSource" ref="dataSourceContent" />  
  67.         "namingStrategy">  
  68.             class="org.hibernate.cfg.ImprovedNamingStrategy" />  
  69.           
  70.         "hibernateProperties">  
  71.               
  72.                 "hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect  
  73.                 "hibernate.show_sql">${hibernate.show_sql}  
  74.                 "hibernate.format_sql">${hibernate.format_sql}  
  75.                 "hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider  
  76.                   
  77.                 "hibernate.cache.provider_configuration_file_resource_path">${hibernate.ehcache_config_file}  
  78.               
  79.           
  80.         "packagesToScan" value="cn.puretext.entity.*" />  
  81.       
  82.   
  83.       
  84.     "transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  85.         "sessionFactory" ref="sessionFactory" />  
  86.       
  87.   
  88.       
  89.       
  90.   
  91.       
  92.     "transactionManager" />  
  93.   

这个时候运行上一篇文章中写好的单元测试DaoTest.java,结果发现还是会出错,错误原因如下: 
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'cn.puretext.unit.service.DaoTest': Autowiring of methods failed; nested exception is org.springframework.beans.factory.BeanCreationException: Could not autowire method: public void org.springframework.test.context.junit4.AbstractTransactionalJUnit4SpringContextTests.setDataSource(javax.sql.DataSource); nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No unique bean of type [javax.sql.DataSource] is defined: expected single matching bean but found 2: [dataSourceContent, dataSourceIndex] 

经过分析,发现是测试类的基类需要注入DataSource,而现在配置了多个DataSource,所以Spring不知道哪个DataSource匹配了,所以需要改写DaoTest.java,如下: 
Java代码   收藏代码
  1. package cn.puretext.unit.service;  
  2.   
  3. import java.util.List;  
  4.   
  5. import javax.annotation.Resource;  
  6. import javax.sql.DataSource;  
  7.   
  8. import org.junit.Test;  
  9. import org.springframework.beans.factory.annotation.Autowired;  
  10. import org.springside.modules.orm.Page;  
  11. import org.springside.modules.test.junit4.SpringTxTestCase;  
  12.   
  13. import cn.puretext.dao.ArticleDao;  
  14. import cn.puretext.entity.web.Article;  
  15.   
  16. public class DaoTest extends SpringTxTestCase {  
  17.     @Autowired  
  18.     private ArticleDao articleDao;  
  19.       
  20.     public ArticleDao getArticleDao() {  
  21.         return articleDao;  
  22.     }  
  23.   
  24.     public void setArticleDao(ArticleDao articleDao) {  
  25.         this.articleDao = articleDao;  
  26.     }  
  27.   
  28.     @Override  
  29.     @Resource(name = "dataSourceContent")  
  30.     public void setDataSource(DataSource dataSource) {  
  31.         // TODO Auto-generated method stub  
  32.         super.setDataSource(dataSource);  
  33.     }  
  34.   
  35.     @Test  
  36.     public void addArticle() {  
  37.         Article article = new Article();  
  38.         article.setSubject("article test");  
  39.         article.setContent("article test");  
  40.         articleDao.save(article);  
  41.     }  
  42.       
  43.     @Test  
  44.     public void pageQuery() {  
  45.         Page
     page = new Page
    ();  
  46.         page.setPageSize(10);  
  47.         page.setPageNo(2);  
  48.         page = articleDao.getAll(page);  
  49.         List
     articles = page.getResult();  
  50.     }  
  51. }  

改变的内容主要为重写了基类中的setDataSource方法,并使用@Resource注解指定使用的DataSource为dataSourceContent。经过修改后,单元测试成功运行。 

第二步,配置多个SessionFactory,配置文件如下: 
Java代码   收藏代码
  1. "1.0" encoding="UTF-8"?>  
  2. "http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  3.     xmlns:jee="http://www.springframework.org/schema/jee" xmlns:tx="http://www.springframework.org/schema/tx"  
  4.     xmlns:context="http://www.springframework.org/schema/context"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"  
  6.     default-lazy-init="true">  
  7.   
  8.     Spring公共配置文件   
  9.   
  10.       
  11.     class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">  
  12.         "systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE" />  
  13.         "ignoreResourceNotFound" value="true" />  
  14.         "locations">  
  15.               
  16.                   
  17.                 classpath*:/application.properties  
  18.                   
  19.                 classpath*:/application.local.properties  
  20.                   
  21.                   
  22.               
  23.           
  24.       
  25.   
  26.       
  27.     package="cn.puretext" />  
  28.   
  29.       
  30.     "dataSourceContent" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">  
  31.           
  32.         "driverClassName" value="com.mysql.jdbc.Driver" />  
  33.         "url" value="${jdbc.urlContent}" />  
  34.         "username" value="${jdbc.username}" />  
  35.         "password" value="${jdbc.password}" />  
  36.   
  37.           
  38.         "initialSize" value="5" />  
  39.         "maxActive" value="100" />  
  40.         "maxIdle" value="30" />  
  41.         "maxWait" value="1000" />  
  42.         "poolPreparedStatements" value="true" />  
  43.         "defaultAutoCommit" value="false" />  
  44.       
  45.     "dataSourceIndex" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">  
  46.           
  47.         "driverClassName" value="com.mysql.jdbc.Driver" />  
  48.         "url" value="${jdbc.urlIndex}" />  
  49.         "username" value="${jdbc.username}" />  
  50.         "password" value="${jdbc.password}" />  
  51.   
  52.           
  53.         "initialSize" value="5" />  
  54.         "maxActive" value="100" />  
  55.         "maxIdle" value="30" />  
  56.         "maxWait" value="1000" />  
  57.         "poolPreparedStatements" value="true" />  
  58.         "defaultAutoCommit" value="false" />  
  59.       
  60.   
  61.       
  62.       
  63.   
  64.       
  65.     "sessionFactoryContent" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">  
  66.         "dataSource" ref="dataSourceContent" />  
  67.         "namingStrategy">  
  68.             class="org.hibernate.cfg.ImprovedNamingStrategy" />  
  69.           
  70.         "hibernateProperties">  
  71.               
  72.                 "hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect  
  73.                 "hibernate.show_sql">${hibernate.show_sql}  
  74.                 "hibernate.format_sql">${hibernate.format_sql}  
  75.                 "hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider  
  76.                   
  77.                 "hibernate.cache.provider_configuration_file_resource_path">${hibernate.ehcache_config_file}  
  78.               
  79.           
  80.         "packagesToScan" value="cn.puretext.entity.*" />  
  81.       
  82.     "sessionFactoryIndex" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">  
  83.         "dataSource" ref="dataSourceIndex" />  
  84.         "namingStrategy">  
  85.             class="org.hibernate.cfg.ImprovedNamingStrategy" />  
  86.           
  87.         "hibernateProperties">  
  88.               
  89.                 "hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect  
  90.                 "hibernate.show_sql">${hibernate.show_sql}  
  91.                 "hibernate.format_sql">${hibernate.format_sql}  
  92.                 "hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider  
  93.                   
  94.                 "hibernate.cache.provider_configuration_file_resource_path">${hibernate.ehcache_config_file}  
  95.               
  96.           
  97.         "packagesToScan" value="cn.puretext.entity.*" />  
  98.       
  99.   
  100.       
  101.     "transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  102.         "sessionFactory" ref="sessionFactoryContent" />  
  103.       
  104.   
  105.       
  106.       
  107.   
  108.       
  109.     "transactionManager" />  
  110.   

运行单元测试,报错,错误代码如下: 
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'cn.puretext.unit.service.DaoTest': Autowiring of fields failed; nested exception is org.springframework.beans.factory.BeanCreationException: Could not autowire field: private cn.puretext.dao.ArticleDao cn.puretext.unit.service.DaoTest.articleDao; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'articleDao': Autowiring of methods failed; nested exception is org.springframework.beans.factory.BeanCreationException: Could not autowire method: public void org.springside.modules.orm.hibernate.SimpleHibernateDao.setSessionFactory(org.hibernate.SessionFactory); nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No unique bean of type [org.hibernate.SessionFactory] is defined: expected single matching bean but found 2: [sessionFactoryContent, sessionFactoryIndex] 

这和上面出现的错误是异曲同工的,只不过这次是ArticleDao类里面不知道注入哪一个SessionFactory,因此,需要修改ArticleDao类,重写setSessionFactory方法,并用@Resource注解指定,如下: 
Java代码   收藏代码
  1. package cn.puretext.dao;  
  2.   
  3.   
  4. import javax.annotation.Resource;  
  5.   
  6. import org.hibernate.SessionFactory;  
  7. import org.springframework.stereotype.Repository;  
  8. import org.springside.modules.orm.hibernate.HibernateDao;  
  9.   
  10. import cn.puretext.entity.web.Article;  
  11.   
  12. @Repository  
  13. public class ArticleDao extends HibernateDao {  
  14.   
  15.     @Override  
  16.     @Resource(name = "sessionFactoryContent")  
  17.     public void setSessionFactory(SessionFactory sessionFactory) {  
  18.         // TODO Auto-generated method stub  
  19.         super.setSessionFactory(sessionFactory);  
  20.     }  
  21.   
  22. }  

运行单元测试,成功。 

第三步、配置多个TransactionManager,如下: 
Java代码   收藏代码
  1. "1.0" encoding="UTF-8"?>  
  2. "http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  3.     xmlns:jee="http://www.springframework.org/schema/jee" xmlns:tx="http://www.springframework.org/schema/tx"  
  4.     xmlns:context="http://www.springframework.org/schema/context"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"  
  6.     default-lazy-init="true">  
  7.   
  8.     Spring公共配置文件   
  9.   
  10.       
  11.     class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">  
  12.         "systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE" />  
  13.         "ignoreResourceNotFound" value="true" />  
  14.         "locations">  
  15.               
  16.                   
  17.                 classpath*:/application.properties  
  18.                   
  19.                 classpath*:/application.local.properties  
  20.                   
  21.                   
  22.               
  23.           
  24.       
  25.   
  26.       
  27.     package="cn.puretext" />  
  28.   
  29.       
  30.     "dataSourceContent" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">  
  31.           
  32.         "driverClassName" value="com.mysql.jdbc.Driver" />  
  33.         "url" value="${jdbc.urlContent}" />  
  34.         "username" value="${jdbc.username}" />  
  35.         "password" value="${jdbc.password}" />  
  36.   
  37.           
  38.         "initialSize" value="5" />  
  39.         "maxActive" value="100" />  
  40.         "maxIdle" value="30" />  
  41.         "maxWait" value="1000" />  
  42.         "poolPreparedStatements" value="true" />  
  43.         "defaultAutoCommit" value="false" />  
  44.       
  45.     "dataSourceIndex" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">  
  46.           
  47.         "driverClassName" value="com.mysql.jdbc.Driver" />  
  48.         "url" value="${jdbc.urlIndex}" />  
  49.         "username" value="${jdbc.username}" />  
  50.         "password" value="${jdbc.password}" />  
  51.   
  52.           
  53.         "initialSize" value="5" />  
  54.         "maxActive" value="100" />  
  55.         "maxIdle" value="30" />  
  56.         "maxWait" value="1000" />  
  57.         "poolPreparedStatements" value="true" />  
  58.         "defaultAutoCommit" value="false" />  
  59.       
  60.   
  61.       
  62.       
  63.   
  64.       
  65.     "sessionFactoryContent" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">  
  66.         "dataSource" ref="dataSourceContent" />  
  67.         "namingStrategy">  
  68.             class="org.hibernate.cfg.ImprovedNamingStrategy" />  
  69.           
  70.         "hibernateProperties">  
  71.               
  72.                 "hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect  
  73.                 "hibernate.show_sql">${hibernate.show_sql}  
  74.                 "hibernate.format_sql">${hibernate.format_sql}  
  75.                 "hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider  
  76.                   
  77.                 "hibernate.cache.provider_configuration_file_resource_path">${hibernate.ehcache_config_file}  
  78.               
  79.           
  80.         "packagesToScan" value="cn.puretext.entity.*" />  
  81.       
  82.     "sessionFactoryIndex" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">  
  83.         "dataSource" ref="dataSourceIndex" />  
  84.         "namingStrategy">  
  85.             class="org.hibernate.cfg.ImprovedNamingStrategy" />  
  86.           
  87.         "hibernateProperties">  
  88.               
  89.                 "hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect  
  90.                 "hibernate.show_sql">${hibernate.show_sql}  
  91.                 "hibernate.format_sql">${hibernate.format_sql}  
  92.                 "hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider  
  93.                   
  94.                 "hibernate.cache.provider_configuration_file_resource_path">${hibernate.ehcache_config_file}  
  95.               
  96.           
  97.         "packagesToScan" value="cn.puretext.entity.*" />  
  98.       
  99.   
  100.       
  101.     "transactionManagerContent" class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  102.         "sessionFactory" ref="sessionFactoryContent" />  
  103.       
  104.     "transactionManagerIndex" class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  105.         "sessionFactory" ref="sessionFactoryIndex" />  
  106.       
  107.   
  108.       
  109.       
  110.   
  111.       
  112.     "transactionManagerContent" />  
  113.   
  114.   
  115. 这个时候运行还是会出错,出错的原因为 org.springframework.beans.factory.NoSuchBeanDefinitionException: No bean named 'transactionManager' is defined,因为该出错信息很短,我也难以找出究竟是哪个地方需要名为“transactionManager”的事务管理器 ,改个名字都不行,看来Spring的自动注入有时候也错综复杂害人不浅。不过,如果把上面的其中一个名字改成“transactionManger”, 另外一个名字不改,运行是成功的,如下:  
  116.   
  117.     "transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  118.         "sessionFactory" ref="sessionFactoryContent" />  
  119.       
  120.     "transactionManagerIndex" class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  121.         "sessionFactory" ref="sessionFactoryIndex" />  
  122.       

这个时候得出结论是,可以配置多个TransactionManager,但是必须有一个的名字是transactionManager。 

第四步、配置多个,如下: 
Java代码   收藏代码
  1.   
  2.     "transactionManager" />  
  3.     "transactionManagerIndex" />  

运行测试,天啦,竟然成功了。和我之前预料的完全不一样,居然在一个配置文件中配置多个一点 问题都没有。那么在使用@Transactional的地方,它真的能够选择正确的事务管理器吗?我不得不写更多的代码来进行测试。那就针对索引数据库中 的表写一个Entity,写一个Dao测试一下吧。 

代码如下: 
Java代码   收藏代码
  1. package cn.puretext.entity.web;  
  2.   
  3. import javax.persistence.Column;  
  4. import javax.persistence.Entity;  
  5. import javax.persistence.Table;  
  6.   
  7. import org.hibernate.annotations.Cache;  
  8. import org.hibernate.annotations.CacheConcurrencyStrategy;  
  9.   
  10. import cn.puretext.entity.IdEntity;  
  11.   
  12. @Entity  
  13. // 表名与类名不相同时重新定义表名.  
  14. @Table(name = "articles")  
  15. // 默认的缓存策略.  
  16. @Cache(usage = CacheConcurrencyStrategy.READ_WRITE)  
  17. public class ArticleIndex extends IdEntity {  
  18.     private String subject;  
  19.     private String webServer;  
  20.   
  21.     public String getSubject() {  
  22.         return subject;  
  23.     }  
  24.   
  25.     public void setSubject(String subject) {  
  26.         this.subject = subject;  
  27.     }  
  28.     @Column(name = "webserver")  
  29.     public String getWebServer() {  
  30.         return webServer;  
  31.     }  
  32.   
  33.     public void setWebServer(String webServer) {  
  34.         this.webServer = webServer;  
  35.     }  
  36. }  

Java代码   收藏代码
  1. package cn.puretext.dao;  
  2.   
  3. import javax.annotation.Resource;  
  4.   
  5. import org.hibernate.SessionFactory;  
  6. import org.springframework.stereotype.Repository;  
  7. import org.springside.modules.orm.hibernate.HibernateDao;  
  8.   
  9. import cn.puretext.entity.web.ArticleIndex;  
  10.   
  11. @Repository  
  12. public class ArticleIndexDao extends HibernateDao {  
  13.     @Override  
  14.     @Resource(name = "sessionFactoryIndex")  
  15.     public void setSessionFactory(SessionFactory sessionFactory) {  
  16.         // TODO Auto-generated method stub  
  17.         super.setSessionFactory(sessionFactory);  
  18.     }  
  19. }  

Java代码   收藏代码
  1. package cn.puretext.unit.service;  
  2.   
  3. import java.util.List;  
  4.   
  5. import javax.annotation.Resource;  
  6. import javax.sql.DataSource;  
  7.   
  8. import org.junit.Test;  
  9. import org.springframework.beans.factory.annotation.Autowired;  
  10. import org.springframework.transaction.annotation.Transactional;  
  11. import org.springside.modules.orm.Page;  
  12. import org.springside.modules.test.junit4.SpringTxTestCase;  
  13.   
  14. import cn.puretext.dao.ArticleDao;  
  15. import cn.puretext.dao.ArticleIndexDao;  
  16. import cn.puretext.entity.web.Article;  
  17. import cn.puretext.entity.web.ArticleIndex;  
  18. import cn.puretext.service.ServiceException;  
  19.   
  20. public class DaoTest extends SpringTxTestCase {  
  21.     @Autowired  
  22.     private ArticleDao articleDao;  
  23.     @Autowired  
  24.     private ArticleIndexDao articleIndexDao;  
  25.       
  26.     public void setArticleIndexDao(ArticleIndexDao articleIndexDao) {  
  27.         this.articleIndexDao = articleIndexDao;  
  28.     }  
  29.   
  30.     public void setArticleDao(ArticleDao articleDao) {  
  31.         this.articleDao = articleDao;  
  32.     }  
  33.   
  34.     @Override  
  35.     @Resource(name = "dataSourceContent")  
  36.     public void setDataSource(DataSource dataSource) {  
  37.         // TODO Auto-generated method stub  
  38.         super.setDataSource(dataSource);  
  39.     }  
  40.   
  41.     @Test  
  42.     @Transactional  
  43.     public void addArticle() {  
  44.         Article article = new Article();  
  45.         article.setSubject("article test");  
  46.         article.setContent("article test");  
  47.         articleDao.save(article);  
  48.     }  
  49.   
  50.     @Test  
  51.     @Transactional  
  52.     public void pageQuery() {  
  53.         Page
     page = new Page
    ();  
  54.         page.setPageSize(10);  
  55.         page.setPageNo(2);  
  56.         page = articleDao.getAll(page);  
  57.         List
     articles = page.getResult();  
  58.     }  
  59.       
  60.     @Test  
  61.     @Transactional  
  62.     public void addIndex() {  
  63.         ArticleIndex articleIndex = new ArticleIndex();  
  64.         articleIndex.setSubject("test");  
  65.         articleIndex.setWebServer("www001");  
  66.         articleIndexDao.save(articleIndex);  
  67.     }  
  68.       
  69.     @Test  
  70.     @Transactional  
  71.     public void addArticleAndAddIndex() {  
  72.         addArticle();  
  73.         addIndex();  
  74.         throw new ServiceException("测试事务回滚");  
  75.     }  
  76. }  

运行测试,结果还是成功的。到目前,发现在一个项目中使用多个TransactionManager可以正常运行,但是有两个问题需要考虑: 
1、为什么必须得有一个TransactionManager名字为transactionManager? 
2、这两个TransactionManager真的能正常工作吗? 
3、OpenSessionInView的问题怎么解决? 

以上的三个问题在单元测试中是不能找出答案的,我只好再去写Action层的代码,期望能够从中得到线索。经过一天艰苦的努力,终于真相大白: 
1、并不是必须有一个TransactionManager的名字为transactionMananger,这只是单元测试在搞鬼,在真实的Web环境 中,无论两个TransactionManager取什么名字都可以,运行不会报错。所以这个答案很明确,是因为单元测试的基类需要一个名为 transactionMananger的事务管理器。 
2、在单元测试中,只能测试Dao类和Entity类能否正常工作,但是由于单元测试结束后事务会自动回滚,不会把数据写入到数据库中,所以没有办法确定 两个TransactionManager能否正常工作。在真实的Web环境中,问题很快就浮出水面,只有一个数据库中有数据,另外一个数据库中没有,经 过调整的位置并对比分析,发现只有放在前面的TransactionMananger的事务 能够正常提交,放在后面的TransactionManager的事务不能提交,所以永远只有一个数据库里面有数据。 
3、如果早一点脱离单元测试而进入真实的Web环境,就会早一点发现OpenSessionInViewFilter的问题,因为只要配置多个 SessionFactory,运行的时候OpenSessionInViewFilter就会报错。为了解决这个问题,我只能去阅读 OpenSessionInViewFilter的源代码,发现它在将Session绑定到线程的时候用的是Map,而且使用 SessionFactory作为Map的key,这就说明在线程中绑定多个Session不会冲突,也进一步说明可以在web.xml中配置多个 OpenSessionInViewFilter。而我也正是通过配置多个OpenSessionInViewFilter来解决问题的。我的 web.xml文件如下: 
Java代码   收藏代码
  1. "1.0" encoding="UTF-8"?>  
  2. "2.4" xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  3.     xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">  
  4.   
  5.     PureText  
  6.       
  7.       
  8.         contextConfigLocation  
  9.         classpath*:/applicationContext*.xml  
  10.       
  11.   
  12.       
  13.       
  14.         encodingFilter  
  15.         class>org.springframework.web.filter.CharacterEncodingFilterclass>  
  16.           
  17.             encoding  
  18.             UTF-8  
  19.           
  20.           
  21.             forceEncoding  
  22.             true  
  23.           
  24.       
  25.   
  26.       
  27.         hibernateOpenSessionInViewFilterContent  
  28.         class>org.springside.modules.orm.hibernate.OpenSessionInViewFilterclass>  
  29.           
  30.             excludeSuffixs  
  31.             js,css,jpg,gif  
  32.           
  33.                 
  34.                sessionFactoryBeanName  
  35.             sessionFactoryContent     
  36.               
  37.       
  38.       
  39.         hibernateOpenSessionInViewFilterIndex  
  40.         class>org.springside.modules.orm.hibernate.OpenSessionInViewFilterclass>  
  41.           
  42.             excludeSuffixs  
  43.             js,css,jpg,gif  
  44.           
  45.                 
  46.                sessionFactoryBeanName  
  47.             sessionFactoryIndex     
  48.               
  49.       
  50.       
  51.       
  52.         springSecurityFilterChain  
  53.         class>org.springframework.web.filter.DelegatingFilterProxyclass>  
  54.       
  55.   
  56.       
  57.       
  58.         struts2Filter  
  59.         class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilterclass>  
  60.       
  61.   
  62.       
  63.         encodingFilter  
  64.         /*  
  65.       
  66.   
  67.   
  68.       
  69.         springSecurityFilterChain  
  70.         /*  
  71.       
  72.       
  73.         hibernateOpenSessionInViewFilterContent  
  74.         /*  
  75.       
  76.       
  77.         hibernateOpenSessionInViewFilterIndex  
  78.         /*  
  79.       
  80.       
  81.         struts2Filter  
  82.         /*  
  83.       
  84.   
  85.       
  86.       
  87.         class>org.springframework.web.context.ContextLoaderListenerclass>  
  88.       
  89.   
  90.       
  91.       
  92.         class>org.springframework.web.util.IntrospectorCleanupListenerclass>  
  93.       
  94.   
  95.       
  96.       
  97.         20  
  98.       
  99.   
  100.       
  101.       
  102.         java.lang.Throwable  
  103.         /common/500.jsp  
  104.       
  105.       
  106.         500  
  107.         /common/500.jsp  
  108.       
  109.       
  110.         404  
  111.         /common/404.jsp  
  112.       
  113.       
  114.         403  
  115.         /common/403.jsp  
  116.       
  117.   

经过上面的分析,发现使用多个TransactionManager是不可行的(这个时候我在想,也许不使用Annotation就可以使用多个 TransactionMananger吧,毕竟Spring的AOP应该是可以把不同的TransactionManager插入到不同的类和方法中, 但是谁愿意走回头路呢?毕竟都已经是@Transactional的年代了),虽然运行不会报错,但是只有一个TransactionManager的事 务能够正常提交。所以测试进入下一步: 

第五步、使用JTATransactionManager 

简单地修改配置文件,使用JTATransactionManager做为事务管理器,配置文件我就不列出来了,运行,结果抱错,错误信息如下: 
org.springframework.beans.factory.BeanCreationException: Error creating bean with name '_filterChainProxy': Initialization of bean failed; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name '_filterChainList': Cannot create inner bean '(inner bean)' of type [org.springframework.security.config.OrderedFilterBeanDefinitionDecorator$OrderedFilterDecorator] while setting bean property 'filters' with key [10]; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name '(inner bean)': Cannot resolve reference to bean 'filterSecurityInterceptor' while setting constructor argument; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'filterSecurityInterceptor' defined in file [D:\Temp\1-PureText\WEB-INF\classes\applicationContext-security.xml]: Cannot resolve reference to bean 'databaseDefinitionSource' while setting bean property 'objectDefinitionSource'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'databaseDefinitionSource': FactoryBean threw exception on object creation; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'org.springframework.transaction.interceptor.TransactionInterceptor#0': Cannot resolve reference to bean 'transactionManager' while setting bean property 'transactionManager'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'transactionManager' defined in file [D:\Temp\1-PureText\WEB-INF\classes\applicationContext.xml]: Invocation of init method failed; nested exception is java.lang.IllegalStateException: No JTA UserTransaction available - specify either 'userTransaction' or 'userTransactionName' or 'transactionManager' or 'transactionManagerName' 

通过分析,发现其中最关键的一句是No JTA UserTransaction available,看来,我们只能进入到第六步,使用GlassFish了。 

第六步、将项目部署到GlassFish中 

将项目简单地部署到GlassFish中之后,项目可以成功运行,没有报错,说明JTA UserTransaction问题解决了,但是检查数据库却发现依然没有数据,看来JTATransactionManager不仅要和应用服务器配合 使用,还要和JNDI数据源一起使用。将数据源的配置修改为JNDI后,问题解决。下面是我的配置文件: 
Java代码   收藏代码
  1. "1.0" encoding="UTF-8"?>  
  2. "http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  3.     xmlns:jee="http://www.springframework.org/schema/jee" xmlns:tx="http://www.springframework.org/schema/tx"  
  4.     xmlns:context="http://www.springframework.org/schema/context"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"  
  6.     default-lazy-init="true">  
  7.   
  8.     Spring公共配置文件   
  9.   
  10.       
  11.     class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">  
  12.         "systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE" />  
  13.         "ignoreResourceNotFound" value="true" />  
  14.         "locations">  
  15.               
  16.                   
  17.                 classpath*:/application.properties  
  18.                   
  19.                 classpath*:/application.local.properties  
  20.                   
  21.                   
  22.               
  23.           
  24.       
  25.   
  26.       
  27.     package="cn.puretext" />  
  28.   
  29.       
  30.     "dataSourceContent" jndi-name="jdbc/dataSourceContent" />  
  31.     "dataSourceIndex" jndi-name="jdbc/dataSourceIndex" />  
  32.   
  33.       
  34.     "sessionFactoryContent" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">  
  35.         "dataSource" ref="dataSourceContent" />  
  36.         "namingStrategy">  
  37.             class="org.hibernate.cfg.ImprovedNamingStrategy" />  
  38.           
  39.         "hibernateProperties">  
  40.               
  41.                 "hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect  
  42.                 "hibernate.show_sql">${hibernate.show_sql}  
  43.                 "hibernate.format_sql">${hibernate.format_sql}  
  44.                 "hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider  
  45.                   
  46.                 "hibernate.cache.provider_configuration_file_resource_path">${hibernate.ehcache_config_file}  
  47.               
  48.           
  49.         "packagesToScan" value="cn.puretext.entity.*" />  
  50.       
  51.     "sessionFactoryIndex" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">  
  52.         "dataSource" ref="dataSourceIndex" />  
  53.         "namingStrategy">  
  54.             class="org.hibernate.cfg.ImprovedNamingStrategy" />  
  55.           
  56.         "hibernateProperties">  
  57.               
  58.                 "hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect  
  59.                 "hibernate.show_sql">${hibernate.show_sql}  
  60.                 "hibernate.format_sql">${hibernate.format_sql}  
  61.                 "hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider  
  62.                   
  63.                 "hibernate.cache.provider_configuration_file_resource_path">${hibernate.ehcache_config_file}  
  64.               
  65.           
  66.         "packagesToScan" value="cn.puretext.entity.*" />  
  67.       
  68.   
  69.       
  70.       
  71.       
  72.       
  73.       
  74.     "transactionManager" class="org.springframework.transaction.jta.JtaTransactionManager" />  
  75.       
  76.   
  77.       
  78.     "transactionManager" />  
  79.       
  80.   

最后,我得出的结论是:要想使用多个数据库,就必须使用JTATransactionMananger,必须使用GlassFish等应用服务器而不是Tomcat,必须使用JNDI来管理dataSource。 

如果一定要使用Tomcat呢? 

这确实是一个难题,但是并不代表着没有解决办法。经过广泛的Google一番之后,终于发现了一个好东东,那就是JOTM,它的全称就是Java Open Transaction Mananger,它的作用就是可以单独提供JTA事务管理的功能,不需要应用服务器。JOTM的使用方法有两种,一种就是把它配置到项目中,和 Spring结合起来使用,另外一种就是把它配置到Tomcat中,这时,Tomcat摇身一变就成了和GlassFish一样的能够提供JTA功能的服 务器了。 

JOTM的官方网站为http://jotm.ow2.org,这是它的新网站,旧网站为http://jotm.objectweb.org。 

我选择了把JOTM 2.0.11整合到Tomcat中的方法进行了测试,结果发现还是不能够正常运行,我使用的是JOTM2.0.11,Tomcat 6.0.20,JKD 6 Update10。看来还得继续折腾下去了。 

另外一个开源的JTA事务管理器是Atomikos,它供了事务管理和连接池,不需要应用服务器支持,其官方网站为http://www.atomikos.com/。有兴趣的朋友可以试试。

你可能感兴趣的:(java)