spring的使用-jdbc事务

Posted liuxuelin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring的使用-jdbc事务相关的知识,希望对你有一定的参考价值。

spring对数据库的操作:jdbcTemplate操作及事务管理

1.jdbcTemplate操作数据库:

      1. 导入jar包(3个)

            spring-jdbc-4.2.4.RELEASE.jar

            spring-tx-4.2.4.RELEASE.jar

            以及jdbc驱动jar包

      2. 创建连接池

            DriverManagerDataSource dataSource = new DriverManagerDataSource();          //为spring内置连接池

      3. 设置连接池参数

            dataSource.setDriverClassName("com.mysql.jdbc.Driver");

            dataSource.setUrl("jdbc:mysql:///springtest");

            dataSource.setUsername("root");

            dataSource.setPassword("abc");

      4. 创建JdbcTemplate对象设置参数

            JdbcTemplate jdbcTemplate=new JdbcTemplate();

            jdbcTemplate.setDataSource(dataSource);

      6. 执行sql语句

            jdbcTemplate.execute("update t_user set name=‘张三‘ where id=1");//execute方法可以执行任意sql

           

      总结:jdbcTemplate类似于jdbcUtils中的queryrunner对象,需要传入一个连接池,可以执行sql语句

 

2.使用spring容器简化上述操作

      将连接池对象DriverManagerDataSource,和temple对象jdbcTemplate都交给spring管理

            1. applicationContext.xml配置:

                  spring内置的连接池DriverManagerDataSource,如下:      //如果要使用c3p0连接池:先导入两个c3p0的两个jar包,在下面的class中传入ComboPooledDataSource全类名,并采用set方法设置响应属性即可

                  <bean id="driverManagerDateSource"    class="org.springframework.jdbc.datasource.DriverManagerDataSource">           

                          <property name="driverClassName" value="com.mysql.jdbc.Driver" /> //采用set方法注入属性,DriverManagerDataSource有对应的set方法

                          <property name="url" value="jdbc:mysql:///springtest" />

                          <property name="username" value="root" />

                          <property name="password" value="abc"></property>

                  </bean>

 

                  配置jdbcTemplate

                  <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">

                        <property name="dataSource" ref="driverManagerDateSource"></property>      //set方法注入属性

                  </bean>

           

            2. 简化后操作:

                  @RunWith(SpringJUnit4ClassRunner.class)

                  @ContextConfiguration(locations = "classpath:applicationContext.xml")

                  public class JdbcTemplateTest2 {

                       @Autowired

                       private JdbcTemplate jdbcTemplate;    //从spring容器中获取对象

 

                       @Test

                       public void test1() {

                       jdbcTemplate.execute("update t_user set sex=‘女‘");   //执行sql语句

                       }

                  }

                      

3.通过JdbcDaoSurport 获取jdbctemplate进一步简化操作           //只需要向jdbcDaoSurport中注入连接池即可

      JdbcDaoSurport类中声明了jdbctemplate和setJdbcTemplate(),和连接池setDataSource()方法。

     

      public class AccountDAOImpl extends JdbcDaoSupport implements IAccountDAO {       //让dao继承JdbcDaoSuppory,需要注入连接池

 

            @Override

            public void accountOut(String outname, double money) {

                  this.getJdbcTemplate().update("update account set money=money-? where name=?", money, outname);       //注入连接池后,直接调用getJdbcTemplate();的方法获取JdbcTemplate对象

            }

      }

     

      如果使用注解方式,注入dataSource的方式:

            @Component

            public class AccountDAOImpl extends JdbcDaoSupport implements IAccountDAO {

                 

                  @Autowired

                  public void setDs(DataSource ds){ //注入时,会在spring容器中查找DataSource的实现类对象,并当作参数传入

                       super.setDataSource(ds);   //调用父类的方法实现连接池注入,注入连接池后,父类会自动创建jdbcTemplate对象

                  }

                  ...

            }

           

4.Jdbctemplate应用--crud操作         

      增删改:update("");           //传入sql语句,与queryrunner一样

            jdbcTemplate.update("update t_user set name=? where id=?", "tom", 1);  //sql语句还可以是insert into 或delete from ...

      查:queryForObject("",class);

 

 

5.引入外部属性文件properties

      1. 创建properties属性文件如:db.properties

     

      2. 在applicationContext.xml中引入properties文件:

            <context:property-placeholder location="classpath:db.properties" />

           

      3. 在配置文件或注解中均可使用${name}方式来获取属性:

            <!-- 创建c3p0连接滨 -->

            <bean id="c3p0DataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">

                  <property name="driverClass" value="${jdbc.driverClass}" />       //获取properties外部文件中的属性并注入

                  <property name="jdbcUrl" value="${jdbc.url}" />

                  <property name="user" value="${jdbc.username}" />      //jdbc.username如果改为usesrname会出错??

                  <property name="password" value="${jdbc.password}" />

            </bean>

     

     

6.事务管理机制(三个接口):

      1. PlatformTransactionManager    --->平台事务管理器,在不同的持久化层,事务代码不一样

            常用的实现类:

                  DataSourceTransactionManager   主要针对于JdbcTemplate开发  MyBatis开发

                  HibernateTransactionManasger    主要针对于Hibernate开发

                  JpaTransactionManager                主要针对于JPA开发

           

            接口中的三个方法:

                  getTransaction();    //开启事务

                  commit();         //提交事务

                  rollback();        //回滚事务

 

      2. TransactionDefinition     --->定义事务的一些相关信息 例如 隔离 传播 超时 只读

            隔离:

                  ISOLATION_DEFUALT              它使用后端数据库的默认隔离级别(spring中选项)

                  ISOLATION_READ_UNCOMMITTED     不能解决问题,会发生脏读 不可重复读 虚读

                  ISOLATION_READ_COMMITTED    可以解决脏读 会产生不可重复读与虚读。

                  ISOLATION_REPEATABLE_READ     可以解决脏读,不可重复读 解决不了虚读

                  ISOLATION_SERIALIZABLE             串行化,可以解决所有问题

 

            传播:  //解决的是两个被事务管理的方法互相调用问题(A调用B)。它与数据库没关系,是程序内部维护的问题

                  PROPAGATION_REQUIRED       默认值 两个操作处于同一个事务,如果之前没有事务,新建一个事务。 两个事务要么同时成功,要么同时失败

                  PROPAGATION_REQUIRES_NEW     B会新建一个事务,两个操作处于不同的事务,A和B互不影响

                  PROPAGATION_NESTED                它是一种嵌套事务,它是使用SavePoint来实现的。A可以影响B,但是B不能影响A。事务回滚时可以回滚到指定的savepoint, 注意:它只对DataSourceTransactionManager有作用

                                                    

                  PROPAGATION_SUPPORTS 支持当前事务,如果不存在,就不使用事务

                  PROPAGATION_MANDATORY 支持当前事务,如果不存在,抛出异常

                  PROPAGATION_NOT_SUPPORTED 以非事务运行,如果有事务存在,挂起当前事务

                  PROPAGATION_NEVER 以非事务运行,如果有事务存在,抛出异常

 

            超时:int TIMEOUT_DEFAULT = -1;     默认值是-1 它使用的是数据库默认的超时时间

            只读:boolean isReadOnly();         它的值有两个true/false,如果选择true一般是在select操作时

           

      3. TransactionStatus     --->定义了事务状态信息,在事务运行过程中,得到某个时间点的状态

           

     

7.spring事务管理-事务管理两种方式

      1.   编码方案 不建议使用,它具有侵入性。在原有的业务代码基础上去添加事务管理代码

     

            @Autowired

            private PlatformTransactionManager transactionManager;      //注入方式获取事务管理器对象

     

            @Override

            public void account(String outname, String inname, double money)  {

                  DefaultTransactionDefinition def = new DefaultTransactionDefinition(); //创建TransactionDefinition实现类对象

                def.setIsolationLevel(TransactionDefinition.ISOLATION_REPEATABLE_READ);      //设置事务相关信息参数

            def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);     

                  def.setReadOnly(false);

                  def.setTimeout(TransactionDefinition.TIMEOUT_DEFAULT);

                 

                  TransactionStatus status = transactionManager.getTransaction(def); //开启事务,需要传入事务定义TransactionDefinition对象

                                                                                                                     //此处得到的是一个事务状态的对象

                  try {

                       // 逻辑代码

                       // 从outname转出money

                       accountDao.accountOut(outname, money);

                       int a = 10 / 0; // 一定会抛出异常

                       // 向inname转入money

                       accountDao.accountIn(inname, money);

                      

                  } catch (Exception e) {

                       //事务回滚

                       transactionManager.rollback(status);   //传入事务状态对象记录状态信息

                  } finally {

                       //提交事务

                       transactionManager.commit(status);

                  }

            }

     

            总结:三个接口的对象相互配合完成事务控制,PlatformTransactionManager对象负责开启、关闭、回滚事务,

                  TransactionDefinition,负责封装事务相关信息,      TransactionStatus负责记录事务状态信息

     

      2.   声明式事务控制,基于AOP对目标进行代理,添加around环绕通知。      //优点:不具有侵入性,不需要修改原来的业务代码

            a.xml 配置文件实现

                  第一步:在applicationContext.xml文件中添加aop与tx的名称空间

                  第二步:配置事务管理器

                       <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

                             <property name="dataSource" ref="c3p0DataSource"/>        //需要注入连接池

                       </bean>

 

                  第三步:配置通知   //Spring为我们提供了一个TransactionInterceptor来完成增强, 对于这个增强,可以使用spring为我们提供的一个标签<tx:advice>来完成操作

                       <tx:advice id="txAdvice" transaction-manager="transactionManager">

                             <tx:attributes>

                                   <tx:method name="account" />

                             </tx:attributes>

                       </tx:advice>

                      

                       <tx:method>中的属性 //相当于定义事务相关信息

                       name:必须的,对哪些方法进行事务控制

                       isolation 可选 设置事务隔离级别 默认是DEFAULT

                       propagation:可选 设置事务传播 默认值 REQUIRED

                       timeout 可选 超时时间 默认值-1

                       read-only 可选 默认值是false 如果不是只读,它可以对insert update delete操作,如果是只读不可以。

                       rollback-for 可选 可以设置一个异常,如果产生这个异常,触发事务回滚

                       no-rolback-for 可选 可以设置一个异常,如果产生这个异常,不会触发事务回滚

                 

                  第四步:配置切面 //使用的是传统的spring的advice,需要使用<aop:advisor>标签配置切面

                       <aop:config>

                             <aop:pointcut

                                   expression="execution(* cn.itheima.service.IAccountService.account(..))" id="txPointcut"/>

                             <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>

                       </aop:config>

 

                      

           

            b.注解实现

                  1. 配置事务管理器       //需要提前配置连接池

                       <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

                             <property name="dataSource" ref="c3p0DataSource"/>        //必须注入dataSource

                       </bean>

                  2. 开启注解事务

                       <tx:annotation-driven transaction-manager="transactionManager"/>            //必须传入事务管理器

 

                  3. 方法或类上使用注解:

                  @Transactional(isolation=Isolation.REPEATABLE_READ,propagation=Propagation.REQUIRED,readOnly=false)      //可以在括号中加入事务信息,也可省略

                       @Override

                       public void account(String outname, String inname, double money) {

                             // 从outname转出money

                             accountDao.accountOut(outname, money);

                             int a = 10 / 0; // 一定会抛出异常

                             // 向inname转入money 

                             accountDao.accountIn(inname, money);

                       }

 

     

this的用法:为当前类的对象的引用,当子类对象调用父类成员时,父类中的this相当于下面的f:

      Z继承F

      F f=new Z();     这里的f就类似于this

      此时this为子类对象的父类引用(多态),当调用getClass()方法时,获取的时子类的字节码对象

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

      

以上是关于spring的使用-jdbc事务的主要内容,如果未能解决你的问题,请参考以下文章

Spring事务JDBC方式下的事务使用示例

JDBC和JMS的Spring事务同步

Spring JDBC 和 事务 的基本使用

如何使用 Guice 在 Spring JDBC 中使用事务

JDBC与Spring事务及事务传播性原理解析-上篇

JDBC与Spring事务及事务传播性原理解析-上篇