到 Google 资讯主页   
EasyJF首页   资料   源码   软件    论坛   网站    
   使用帮助    
    该信息为本站MyRSS系统缓存内容,部分图片及附件有可能无法正常使用.easyjf.comwww.matrix.org.cn无关,不对该信息负责.通过http://www.matrix.org.cn//resource/article/1/1339.html访问该信息的原始内容.
页面功能  【加入收藏】 【推荐给朋友】 【字体:  】 【关闭】   
Learn Spring in spring(四)
作者:bromon 来源:www.matrix.org.cn  发布时间:2006-02-22 17:52:32.843

四、Spring中的事务控制

        Spring和EJB一样,提供了两种事务管理方式:编程式和声明式。在考试系统中我们将使用声明式的事务管理,这是spring推荐的做法。使用这种方式可以体验到spring的强大便捷,而且我们无须在Dao类中编写任何特殊的代码,只需要通过配置文件就可以让普通的java类加载到事务管理中,这个意义是很重大的。

        Spring中进行事务管理的通常方式是利用AOP(面向切片编程)的方式,为普通java类封装事务控制,它是通过动态代理实现的,由于接口是延迟实例化的,spring在这段时间内通过拦截器,加载事务切片。原理就是这样,具体细节请参考jdk中有关动态代理的文档。本文主要讲解如何在spring中进行事务控制。

        动态代理的一个重要特征是,它是针对接口的,所以我们的dao要通过动态代理来让spring接管事务,就必须在dao前面抽象出一个接口,当然如果没有这样的接口,那么spring会使用CGLIB来解决问题,但这不是spring推荐的方式,我们也不做讨论。

        参照前面的例子,我们为StudentManager.java定义一个接口,它的内容如下:

/*
* 创建日期 2005-3-25
*/
package org.bromon.spring.examer.student;

import java.util.List;

import org.bromon.spring.examer.pojo.Student;

/**
* @author Bromon
*/
public interface StudentManagerInterface
{
    public void add(Student s);
    public void del(Student s);
    public void update(Student s);
    
    public List loadAll();
    public Student loadById(int id);
}

StudentManager也应该做出修改,实现该接口:

public class StudentManager extends HibernateDaoSupport implements StudentManagerInterface
        
现在需要修改配置文件,用于定义Hibrenate适用的事务管理器,并且把sessionFactory注入进去,同时还需要通过注册一个DefaultTransactionAttribute对象,来指出事务策略。其中sessionFactory的定义已经在本文的第三章中说明。

        首先定义一个Hibernate的事务管理器,让它来管理sessionFactory:
<bean id="transactionManager" class="org.springframework.orm.hibernate.HibernateTransactionManager">
                <property name="sessionFactory">
                        <ref bean="sessionFactory"/>
                </property>
</bean>

下面定义事务管理策略,我们希望把策略定义在方法这个级别上,提供最大的灵活性,本例中将add方法定义为:PROPAGATION_REQUIRES_NEW,这可以保证它将始终运行在一个事务中。

<bean id="transactionAttributeSource" class="org.springframework.transaction.interceptor.NameMatchTransactionAttributeSource">
                <property name="properties">
                        <props>
                                <prop key="add">
                                        PROPAGATION_REQUIRES_NEW
                                </prop>
                        </props>
                </property>
        </bean>

我们不仅可以为add方法定义事务策略,还可以定义事务隔离程度和回滚策略,他们以逗号隔开,比如我们的add事务可以定义为:

<prop key="add">
        PROPAGATION_REQUIRES_NEW,-ExamerException
</prop
>

        这个事务策略表示add方法将会独占一个事务,当事务过程中产生ExamerException异常,事务会回滚。

        Add/update/del都是写入方法,对于select(读取)方法,我们可以指定较为复杂的事务策略,比如对于loadAll()方法:

        
<prop key=”loadAll”>
                PROPAGATION_SUPPORTS,ISOLATION_READ_COMMITED,readOnly
        </prop>


        该事务的含义为,loadAll方法支持事务,不会读去位提交的数据,它的数据为只读(可提高执行速度)。

        如你所见,我们的StudentManagerInterface接口中还有一个loadById(int id)方法,也许我们将来还会有很多的loadByXXXX的方法,难道要意义为他们指定事务策略?太烦人了,他们应该和loadAll()一样,所以我们可以使用通配符,定义所有的loadXXXX方法:

        
<prop key=”load*”>
                PROPAGATION_SUPPORTS,ISOLATION_READ_COMMITED,readOnly
        </prop>


        现在可以定义事务管理器:
<bean id="studentManager" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
                <property name="target">
                        <ref bean="studentManager"/>
                </property>
                <property name="transactionManager">
                        <ref bean="transactionManager"/>
                </property>
                <property name="transactionAttributeSource">
                        <ref bean="transactionAttributeSource"/>
                </property>
</bean>


        这个bean的外观是一个接口(StudentManagerInterface),我们指出了它的具体实现(studentManager),而且为它绑定了事务策略。在客户端使用的时候,获得对象是StudentManagerInterface,所有的操作都是针对这个接口的。测试代码并没有改变,我们虽然修改了很多地方,加入了事务控制,但是客户端并没有受到影响,这也体现了spring的一些优势。测试代码如下:

        
public void testAdd() 
    {
        ApplicationContext ctx=new ClassPathXmlApplicationContext("springConfig.xml");
        StudentManager sm=(StudentManager)ctx.getBean("studentManager");
        
        Student s=new Student();
        s.setId(1);
        s.setName("bromon");
        s.setPassword("123");
        s.setGrade(1);
        s.setSex(0);
        
        sm.add(s);
}


        通过以上的代码可以看出,spring可以简单的把普通的java class纳入事务管理,声明性的事务操作起来也很容易。有了spring之后,声明性事务不再是EJB独有,我们不必为了获得声明性事务的功能而去忍受EJB带来的种种不便。

        我所使用的mysql是不支持事务的,你可以更换使用PostgreSQL,有了spring+hibernate,更换db并不像以前那样恐怖了,步骤很简单:

1、        添加PostgreSQL的jdbc驱动
2、        修改dataSource配置,包括驱动名称、url、帐号、密码
3、        修改sessionFactory的数据库dailet为net.sf.hibernate.dialect.PostgreSQLDialect
4、        修改hbm.xml中的主键生成策略为increment

所有的修改都在配置文件中完成,业务代码不需要任何修改,我很满意,How about u?

附A  pring中的所有事务策略

        PROPAGATION_MANDATORY
        PROPAGATION_NESTED                        
        PROPAGATION_NEVER                        
        PROPAGATION_NOT_SUPPORTED
        PROPAGATION_REQUIRED
        PROPAGATION_REQUIRED_NEW
        PROPAGATION_SUPPORTS

附B                Spring中所有的隔离策略:

        ISOLATION_DEFAULT
        ISOLATION_READ_UNCOMMITED
        ISOLATION_COMMITED
        ISOLATION_REPEATABLE_READ
        ISOLATION_SERIALIZABLE

下一篇:spring中的jms

 
相关文章
 
页面功能  【加入收藏】 【推荐给朋友】 【字体:  】 【关闭】   


EasyJF.com 2006 隐私政策 使用EasyJF前必读