Spring JdbcTemplate 与 事务管理 学习
Spring的JDBC框架能够承担资源管理和异常处理的工作,从而简化我们的JDBC代码,
让我们只需编写从数据库读写数据所必需的代码。Spring把数据访问的样板代码隐藏到模板类之下,
结合Spring的事务管理,可以大大简化我们的代码.
Spring提供了3个模板类:
JdbcTemplate:Spring里最基本的JDBC模板,利用JDBC和简单的索引参数查询提供对数据库的简单访问。
NamedParameterJdbcTemplate:能够在执行查询时把值绑定到SQL里的命名参数,而不是使用索引参数。
SimpleJdbcTemplate:利用Java 5的特性,比如自动装箱、通用(generic)和可变参数列表来简化JDBC模板的使用。
具体使用哪个模板基本上取决于个人喜好。
使用Spring的JdbcTemplate来实现简单的增删改查,首先建立测试数据表person
create table person(
id int not null primary key auto_increment,
name varchar(20) not null
)
导入依赖的jar包,由于测试中数据源使用的是dbcp数据源,需要以下jar包支持:
commons-logging.jar
commons-pool.jar
commons-dbcp.jar
同时还必须导入数据库驱动jar包:mysql-connector-java-3.1.8-bin.jar
建立实体bean
Person.java
- package com.royzhou.jdbc;
- public class PersonBean {
- private int id;
- private String name;
- public PersonBean() {
- }
- public PersonBean(String name) {
- this.name = name;
- }
- public PersonBean(int id, String name) {
- this.id = id;
- this.name = name;
- }
- public int getId() {
- return id;
- }
- public void setId(int id) {
- this.id = id;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String toString() {
- return this.id + ":" + this.name;
- }
- }
接口类:
PersonService.java
- package com.royzhou.jdbc;
- import java.util.List;
- public interface PersonService {
- public void addPerson(PersonBean person);
- public void updatePerson(PersonBean person);
- public void deletePerson(int id);
- public PersonBean queryPerson(int id);
- public List<PersonBean> queryPersons();
- }
实现类:
PersonServiceImpl.java
- package com.royzhou.jdbc;
- import java.util.List;
- import javax.sql.DataSource;
- import java.sql.Types;
- import org.springframework.jdbc.core.JdbcTemplate;
- public class PersonServiceImpl implements PersonService {
- private JdbcTemplate jdbcTemplate;
- /**
- * 通过Spring容器注入datasource
- * 实例化JdbcTemplate,该类为主要操作数据库的类
- * @param ds
- */
- public void setDataSource(DataSource ds) {
- this.jdbcTemplate = new JdbcTemplate(ds);
- }
- public void addPerson(PersonBean person) {
- /**
- * 第一个参数为执行sql
- * 第二个参数为参数数据
- * 第三个参数为参数类型
- */
- jdbcTemplate.update("insert into person values(null,?)", new Object[]{person.getName()}, new int[]{Types.VARCHAR});
- }
- public void deletePerson(int id) {
- jdbcTemplate.update("delete from person where id = ?", new Object[]{id}, new int[]{Types.INTEGER});
- }
- public PersonBean queryPerson(int id) {
- /**
- * new PersonRowMapper()是一个实现RowMapper接口的类,
- * 执行回调,实现mapRow()方法将rs对象转换成PersonBean对象返回
- */
- PersonBean pb = (PersonBean) jdbcTemplate.queryForObject("select id,name from person where id = ?", new Object[]{id}, new PersonRowMapper());
- return pb;
- }
- @SuppressWarnings("unchecked")
- public List<PersonBean> queryPersons() {
- List<PersonBean> pbs = (List<PersonBean>) jdbcTemplate.query("select id,name from person", new PersonRowMapper());
- return pbs;
- }
- public void updatePerson(PersonBean person) {
- jdbcTemplate.update("update person set name = ? where id = ?", new Object[]{person.getName(), person.getId()}, new int[]{Types.VARCHAR, Types.INTEGER});
- }
- }
PersonRowMapper.java
- package com.royzhou.jdbc;
- import java.sql.ResultSet;
- import java.sql.SQLException;
- import org.springframework.jdbc.core.RowMapper;
- public class PersonRowMapper implements RowMapper {
- //默认已经执行rs.next(),可以直接取数据
- public Object mapRow(ResultSet rs, int index) throws SQLException {
- PersonBean pb = new PersonBean(rs.getInt("id"),rs.getString("name"));
- return pb;
- }
- }
我们需要在bean.xml中配置DataSource,并且将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:context="http://www.springframework.org/schema/context"
- xmlns:aop="http://www.springframework.org/schema/aop"
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
- <context:property-placeholder location="classpath:jdbc.properties"/>
- <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
- <property name="driverClassName" value="${driverClassName}"/>
- <property name="url" value="${url}"/>
- <property name="username" value="${username}"/>
- <property name="password" value="${password}"/>
- <!-- 连接池启动时的初始值 -->
- <property name="initialSize" value="${initialSize}"/>
- <!-- 连接池的最大值 -->
- <property name="maxActive" value="${maxActive}"/>
- <!-- 最大空闲值.当经过一个高峰时间后,连接池可以慢慢将已经用不到的连接慢慢释放一部分,一直减少到maxIdle为止 -->
- <property name="maxIdle" value="${maxIdle}"/>
- <!-- 最小空闲值.当空闲的连接数少于阀值时,连接池就会预申请去一些连接,以免洪峰来时来不及申请 -->
- <property name="minIdle" value="${minIdle}"/>
- </bean>
- </beans>
jdbc.properties
- driverClassName=org.gjt.mm.mysql.Driver
- url=jdbc:mysql://localhost:3306/royzhou?useUnicode=true&characterEncoding=UTF-8
- username=root
- password=123456
- initialSize=1
- maxActive=500
- maxIdle=2
- minIdle=1
编写我们的测试类:TestJdbcTemplate.java
- package com.royzhou.jdbc;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
- public class TestJdbcTemplate {
- public static void main(String[] args) {
- ApplicationContext ctx = new ClassPathXmlApplicationContext("bean.xml");
- PersonService ps = (PersonService)ctx.getBean("personService");
- ps.addPerson(new PersonBean("royzhou"));
- PersonBean pb = ps.queryPerson(1);
- System.out.println(pb);
- pb.setName("haha");
- ps.updatePerson(pb);
- pb = ps.queryPerson(1);
- System.out.println(pb);
- ps.deletePerson(1);
- pb = ps.queryPerson(1);
- System.out.println(pb);
- }
- }
上面代码先插入一条记录,然后修改,之后删除,运行之后出现异常,异常信息:
EmptyResultDataAccessException: Incorrect result size: expected 1, actual 0
难道Spring的queryForObject在查找不到记录的时候会抛出异常,看了一下Spring的源代码 发现确实如此:
- public Object queryForObject(String sql, Object[] args, int[] argTypes, RowMapper rowMapper) throws DataAccessException {
- List results = (List) query(sql, args, argTypes, new RowMapperResultSetExtractor(rowMapper, 1));
- return DataAccessUtils.requiredUniqueResult(results);
- }
- public Object queryForObject(String sql, Object[] args, RowMapper rowMapper) throws DataAccessException {
- List results = (List) query(sql, args, new RowMapperResultSetExtractor(rowMapper, 1));
- return DataAccessUtils.requiredUniqueResult(results);
- }
- public Object queryForObject(String sql, RowMapper rowMapper) throws DataAccessException {
- List results = query(sql, rowMapper);
- return DataAccessUtils.requiredUniqueResult(results);
- }
- public static Object requiredUniqueResult(Collection results) throws IncorrectResultSizeDataAccessException {
- int size = (results != null ? results.size() : 0);
- if (size == 0) {
- throw new EmptyResultDataAccessException(1); // 问题在这里
- }
- if (!CollectionUtils.hasUniqueObject(results)) {
- throw new IncorrectResultSizeDataAccessException(1, size);
- }
- return results.iterator().next();
- }
发现当查找不到记录是,requiredUniqueResult方法做了判断,抛出异常, 想不明白为什么Spring要在这里做这样的判断,为啥不返回null????
重新修改PersonServiceImple类,把queryPerson方法改为使用列表查询的方式再去根据index取
PersonServiceImpl.java
- package com.royzhou.jdbc;
- import java.util.List;
- import javax.sql.DataSource;
- import java.sql.Types;
- import org.springframework.jdbc.core.JdbcTemplate;
- public class PersonServiceImpl implements PersonService {
- private JdbcTemplate jdbcTemplate;
- /**
- * 通过Spring容器注入datasource
- * 实例化JdbcTemplate,该类为主要操作数据库的类
- * @param ds
- */
- public void setDataSource(DataSource ds) {
- this.jdbcTemplate = new JdbcTemplate(ds);
- }
- public void addPerson(PersonBean person) {
- /**
- * 第一个参数为执行sql
- * 第二个参数为参数数据
- * 第三个参数为参数类型
- */
- jdbcTemplate.update("insert into person values(null,?)", new Object[]{person.getName()}, new int[]{Types.VARCHAR});
- }
- public void deletePerson(int id) {
- jdbcTemplate.update("delete from person where id = ?", new Object[]{id}, new int[]{Types.INTEGER});
- }
- @SuppressWarnings("unchecked")
- public PersonBean queryPerson(int id) {
- /**
- * new PersonRowMapper()是一个实现RowMapper接口的类,
- * 执行回调,实现mapRow()方法将rs对象转换成PersonBean对象返回
- */
- List<PersonBean> pbs = (List<PersonBean>)jdbcTemplate.query("select id,name from person where id = ?", new Object[]{id}, new PersonRowMapper());
- PersonBean pb = null;
- if(pbs.size()>0) {
- pb = pbs.get(0);
- }
- return pb;
- }
- @SuppressWarnings("unchecked")
- public List<PersonBean> queryPersons() {
- List<PersonBean> pbs = (List<PersonBean>) jdbcTemplate.query("select id,name from person", new PersonRowMapper());
- return pbs;
- }
- public void updatePerson(PersonBean person) {
- jdbcTemplate.update("update person set name = ? where id = ?", new Object[]{person.getName(), person.getId()}, new int[]{Types.VARCHAR, Types.INTEGER});
- }
- }
再次运行测试类,输出:
1:royzhou
1:haha
null
得到预期的结果.
从上面代码可以看出,使用Spring提供的JDBCTemplate类很大程度减少了我们的代码量,
比起以前我们写JDBC操作,需要先获取Connection,然后是PreparedStatement,再到Result,
使用Spring JDBCTemplate写出来的代码看起来更加简洁,开发效率也比较快.
在数据库的操作中,事务是一个重要的概念,举个例子:
大概每个人都有转账的经历。当我们从A帐户向B帐户转100元后,银行的系统会从A帐户上扣除100而在B帐户上加100,这是一般的正常现象。
但是一旦系统出错了怎么办呢,这里我们假设可能会发生两种情况:
(1)A帐户上少了100元,但是B帐户却没有多100元。
(2)B帐户多了100元钱,但是A帐户上却没有被扣钱。
这种错误一旦发生就等于出了大事,那么再假如一下,你要转账的是1亿呢?
所以上面的两种情况分别是你和银行不愿意看到的,因为谁都不希望出错。那么有没有什么方法保证一旦A帐户上没有被扣钱而B帐户上也没有被加钱;
或者A帐户扣了100元而B帐户准确无误的加上100元呢。也就是说要么转账顺利的成功进行,要么不转账呢?可以,这就是数据库事务机制所要起到的作用和做的事情。
Spring对事务的管理有丰富的支持,Spring提供了编程式配置事务和声明式配置事务:
声明式事务有以下两种方式
一种是使用Annotation注解的方式(官方推荐)
一种是基于Xml的方式
采用任何一种方式我们都需要在我们的bean.xml中添加事务支持:
- <?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:context="http://www.springframework.org/schema/context"
- xmlns:aop="http://www.springframework.org/schema/aop"
- 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.5.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
- <context:property-placeholder location="classpath:jdbc.properties" />
- <bean id="dataSource"
- class="org.apache.commons.dbcp.BasicDataSource"
- destroy-method="close">
- <property name="driverClassName" value="${driverClassName}" />
- <property name="url" value="${url}" />
- <property name="username" value="${username}" />
- <property name="password" value="${password}" />
- <!-- 连接池启动时的初始值 -->
- <property name="initialSize" value="${initialSize}" />
- <!-- 连接池的最大值 -->
- <property name="maxActive" value="${maxActive}" />
- <!-- 最大空闲值.当经过一个高峰时间后,连接池可以慢慢将已经用不到的连接慢慢释放一部分,一直减少到maxIdle为止 -->
- <property name="maxIdle" value="${maxIdle}" />
- <!-- 最小空闲值.当空闲的连接数少于阀值时,连接池就会预申请去一些连接,以免洪峰来时来不及申请 -->
- <property name="minIdle" value="${minIdle}" />
- </bean>
- <bean id="personService"
- class="com.royzhou.jdbc.PersonServiceImpl">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <bean id="txManager"
- class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
- <property name="dataSource" ref="dataSource" />
- </bean>
- <tx:annotation-driven transaction-manager="txManager" />
- </beans>
<tx:annotation-driven transaction-manager="txManager" /> 这句话的作用是注册事务注解处理器
定义好配置文件后我们只需要在我们的类上加上注解@Transactional,就可以指定这个类需要受Spring的事务管理
默认Spring为每个方法开启一个事务,如果方法发生运行期错误unchecked(RuntimeException),事务会进行回滚
如果发生checked Exception,事务不进行回滚.
例如在下面的例子中,我们为PersonServiceImpl添加事务支持.
- package com.royzhou.jdbc;
- import java.util.List;
- import javax.sql.DataSource;
- import java.sql.Types;
- import org.springframework.jdbc.core.JdbcTemplate;
- import org.springframework.transaction.annotation.Transactional;
- @Transactional
- public class PersonServiceImpl implements PersonService {
- private JdbcTemplate jdbcTemplate;
- /**
- * 通过Spring容器注入datasource
- * 实例化JdbcTemplate,该类为主要操作数据库的类
- * @param ds
- */
- public void setDataSource(DataSource ds) {
- this.jdbcTemplate = new JdbcTemplate(ds);
- }
- public void addPerson(PersonBean person) {
- /**
- * 第一个参数为执行sql
- * 第二个参数为参数数据
- * 第三个参数为参数类型
- */
- jdbcTemplate.update("insert into person values(null,?)", new Object[]{person.getName()}, new int[]{Types.VARCHAR});
- throw new RuntimeException("运行期例外");
- }
- public void deletePerson(int id) {
- jdbcTemplate.update("delete from person where id = ?", new Object[]{id}, new int[]{Types.INTEGER});
- }
- @SuppressWarnings("unchecked")
- public PersonBean queryPerson(int id) {
- /**
- * new PersonRowMapper()是一个实现RowMapper接口的类,
- * 执行回调,实现mapRow()方法将rs对象转换成PersonBean对象返回
- */
- List<PersonBean> pbs = (List<PersonBean>)jdbcTemplate.query("select id,name from person where id = ?", new Object[]{id}, new PersonRowMapper());
- PersonBean pb = null;
- if(pbs.size()>0) {
- pb = pbs.get(0);
- }
- return pb;
- }
- @SuppressWarnings("unchecked")
- public List<PersonBean> queryPersons() {
- List<PersonBean> pbs = (List<PersonBean>) jdbcTemplate.query("select id,name from person", new PersonRowMapper());
- return pbs;
- }
- public void updatePerson(PersonBean person) {
- jdbcTemplate.update("update person set name = ? where id = ?", new Object[]{person.getName(), person.getId()}, new int[]{Types.VARCHAR, Types.INTEGER});
- }
- }
在addPerson方法中我们抛出了一个运行期例外,以此来检查Spring的事务管理.
编写测试类运行:
- package com.royzhou.jdbc;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
- public class TestJdbcTemplate {
- public static void main(String[] args) {
- ApplicationContext ctx = new ClassPathXmlApplicationContext("bean.xml");
- PersonService ps = (PersonService)ctx.getBean("personService");
- ps.addPerson(new PersonBean("royzhou"));
- }
- }
运行测试类,后台输出异常:java.lang.RuntimeException: 运行期例外
查看数据库发现数据没有插入,说明事务进行了回滚.
再次测试修改为抛出checked Exception
- package com.royzhou.jdbc;
- import java.util.List;
- import javax.sql.DataSource;
- import java.sql.Types;
- import org.springframework.jdbc.core.JdbcTemplate;
- import org.springframework.transaction.annotation.Transactional;
- @Transactional
- public class PersonServiceImpl implements PersonService {
- private JdbcTemplate jdbcTemplate;
- /**
- * 通过Spring容器注入datasource
- * 实例化JdbcTemplate,该类为主要操作数据库的类
- * @param ds
- */
- public void setDataSource(DataSource ds) {
- this.jdbcTemplate = new JdbcTemplate(ds);
- }
- public void addPerson(PersonBean person) throws Exception {
- /**
- * 第一个参数为执行sql
- * 第二个参数为参数数据
- * 第三个参数为参数类型
- */
- jdbcTemplate.update("insert into person values(null,?)", new Object[]{person.getName()}, new int[]{Types.VARCHAR});
- throw new Exception("checked 例外");
- }
- public void deletePerson(int id) {
- jdbcTemplate.update("delete from person where id = ?", new Object[]{id}, new int[]{Types.INTEGER});
- }
- @SuppressWarnings("unchecked")
- public PersonBean queryPerson(int id) {
- /**
- * new PersonRowMapper()是一个实现RowMapper接口的类,
- * 执行回调,实现mapRow()方法将rs对象转换成PersonBean对象返回
- */
- List<PersonBean> pbs = (List<PersonBean>)jdbcTemplate.query("select id,name from person where id = ?", new Object[]{id}, new PersonRowMapper());
- PersonBean pb = null;
- if(pbs.size()>0) {
- pb = pbs.get(0);
- }
- return pb;
- }
- @SuppressWarnings("unchecked")
- public List<PersonBean> queryPersons() {
- List<PersonBean> pbs = (List<PersonBean>) jdbcTemplate.query("select id,name from person", new PersonRowMapper());
- return pbs;
- }
- public void updatePerson(PersonBean person) {
- jdbcTemplate.update("update person set name = ? where id = ?", new Object[]{person.getName(), person.getId()}, new int[]{Types.VARCHAR, Types.INTEGER});
- }
- }
后台输出异常:java.lang.Exception: checked 例外
查看数据库发现数据插入,说明事务没有进行了回滚.
说明了Spring的事务支持默认只对运行期异常(RuntimeException)进行回滚,这里可能有个疑问,我们执行sql操作的时候会发生sql异常,不属于运行期异常,那Spring是怎么进行事务回滚的呢 ????
查看了一下JdbcTemplate的源代码发现,JdbcTemplate的处理方法如下:
- public int[] batchUpdate(String sql, final BatchPreparedStatementSetter pss) throws DataAccessException {
- if (logger.isDebugEnabled()) {
- logger.debug("Executing SQL batch update [" + sql + "]");
- }
- return (int[]) execute(sql, new PreparedStatementCallback() {
- public Object doInPreparedStatement(PreparedStatement ps) throws SQLException {
- try {
- int batchSize = pss.getBatchSize();
- if (JdbcUtils.supportsBatchUpdates(ps.getConnection())) {
- for (int i = 0; i < batchSize; i++) {
- pss.setValues(ps, i);
- ps.addBatch();
- }
- return ps.executeBatch();
- }
- else {
- int[] rowsAffected = new int[batchSize];
- for (int i = 0; i < batchSize; i++) {
- pss.setValues(ps, i);
- rowsAffected[i] = ps.executeUpdate();
- }
- return rowsAffected;
- }
- }
- finally {
- if (pss instanceof ParameterDisposer) {
- ((ParameterDisposer) pss).cleanupParameters();
- }
- }
- }
- });
- }
在代码中捕获了SQLException然后抛出一个org.springframework.dao.DataAcceddException,该异常继承自org.springframework.core.NestedRuntimeException,NestedRuntimeException
是一个继承自RuntimeException的抽象类,Spring jdbcTemplate处理发生异常处理后抛出来得异常基本上都会继承NestedRuntimeException,看完之后才确信了Spring默认只对RuntimeException进行回滚
当然我们可可以修改Spring的默认配置,当发生RuntimeException我们也可以不让他进行事务回滚
只需要加上一个@Transactional(noRollbackFor=RuntimeException.class)
注意@Transactional只能针对public属性范围内的方法添加
- package com.royzhou.jdbc;
- import java.util.List;
- import javax.sql.DataSource;
- import java.sql.Types;
- import org.springframework.jdbc.core.JdbcTemplate;
- import org.springframework.transaction.annotation.Transactional;
- @Transactional
- public class PersonServiceImpl implements PersonService {
- private JdbcTemplate jdbcTemplate;
- /**
- * 通过Spring容器注入datasource
- * 实例化JdbcTemplate,该类为主要操作数据库的类
- * @param ds
- */
- public void setDataSource(DataSource ds) {
- this.jdbcTemplate = new JdbcTemplate(ds);
- }
- @Transactional(noRollbackFor=RuntimeException.class)
- public void addPerson(PersonBean person) {
- /**
- * 第一个参数为执行sql
- * 第二个参数为参数数据
- * 第三个参数为参数类型
- */
- jdbcTemplate.update("insert into person values(null,?)", new Object[]{person.getName()}, new int[]{Types.VARCHAR});
- throw new RuntimeException("运行期例外");
- }
- public void deletePerson(int id) {
- jdbcTemplate.update("delete from person where id = ?", new Object[]{id}, new int[]{Types.INTEGER});
- }
- @SuppressWarnings("unchecked")
- public PersonBean queryPerson(int id) {
- /**
- * new PersonRowMapper()是一个实现RowMapper接口的类,
- * 执行回调,实现mapRow()方法将rs对象转换成PersonBean对象返回
- */
- List<PersonBean> pbs = (List<PersonBean>)jdbcTemplate.query("select id,name from person where id = ?", new Object[]{id}, new PersonRowMapper());
- PersonBean pb = null;
- if(pbs.size()>0) {
- pb = pbs.get(0);
- }
- return pb;
- }
- @SuppressWarnings("unchecked")
- public List<PersonBean> queryPersons() {
- List<PersonBean> pbs = (List<PersonBean>) jdbcTemplate.query("select id,name from person", new PersonRowMapper());
- return pbs;
- }
- public void updatePerson(PersonBean person) {
- jdbcTemplate.update("update person set name = ? where id = ?", new Object[]{person.getName(), person.getId()}, new int[]{Types.VARCHAR, Types.INTEGER});
- }
- }
运行测试类:
- package com.royzhou.jdbc;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
- public class TestJdbcTemplate {
- public static void main(String[] args) {
- ApplicationContext ctx = new ClassPathXmlApplicationContext("bean.xml");
- PersonService ps = (PersonService)ctx.getBean("personService");
- ps.addPerson(new PersonBean("royzhou"));
- }
- }
后台抛出异常,查看数据库,记录插入进去了,说明我们配置事务不对RuntimeException回滚生效了.
既然可以配置不对RuntimeException回滚,那我们也可以配置对Exception进行回滚,主要用到的是
@Transactional(rollbackFor=Exception.class)
对于一些查询工作,因为不需要配置事务支持,我们配置事务的传播属性:
@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true)
readOnly=true表示事务中不允许存在更新操作.
关于事务的传播属性有下面几种配置:
REQUIRED:业务方法需要在一个事务中运行,如果方法运行时,已经处于一个事务中,那么加入到该事务中,否则自己创建一个新的事务.(Spring默认的事务传播属性)
NOT_SUPPORTED:声明方法不需要事务,如果方法没有关联到一个事务,容器不会为它开启事务,如果方法在一个事务中被调用,该事务被挂起,在方法调用结束后,原先的事务便会恢复执行
REQUIRESNEW:不管是否存在事务,业务方法总会为自己发起一个新的事务,如果方法运行时已经存在一个事务,则该事务会被挂起,新的事务被创建,知道方法执行结束,新事务才结束,原先的事务才恢复执行.
MANDATORY:指定业务方法只能在一个已经存在的事务中执行,业务方法不能自己发起事务,如果业务方法没有在事务的环境下调用,则容器会抛出异常
SUPPORTS:如果业务方法在事务中被调用,则成为事务中的一部分,如果没有在事务中调用,则在没有事务的环境下执行
NEVER:指定业务方法绝对不能在事务范围内运行,否则会抛出异常.
NESTED:如果业务方法运行时已经存在一个事务,则新建一个嵌套的事务,该事务可以有多个回滚点,如果没有事务,则按REQUIRED属性执行. 注意:业务方法内部事务的回滚不会对外部事务造成影响,但是外部事务的回滚会影响内部事务
关于使用注解的方式来配置事务就到这里,
我们还可以使用另外一种方式实现事务的管理,通过xml文件的配置,主要通过AOP技术实现:
首先把我们的业务类的注解去掉:
- package com.royzhou.jdbc;
- import java.util.List;
- import javax.sql.DataSource;
- import java.sql.Types;
- import org.springframework.jdbc.core.JdbcTemplate;
- public class PersonServiceImpl implements PersonService {
- private JdbcTemplate jdbcTemplate;
- /**
- * 通过Spring容器注入datasource
- * 实例化JdbcTemplate,该类为主要操作数据库的类
- * @param ds
- */
- public void setDataSource(DataSource ds) {
- this.jdbcTemplate = new JdbcTemplate(ds);
- }
- public void addPerson(PersonBean person) {
- /**
- * 第一个参数为执行sql
- * 第二个参数为参数数据
- * 第三个参数为参数类型
- */
- jdbcTemplate.update("insert into person values(null,?)", new Object[]{person.getName()}, new int[]{Types.VARCHAR});
- throw new RuntimeException("运行期例外");
- }
- public void deletePerson(int id) {
- jdbcTemplate.update("delete from person where id = ?", new Object[]{id}, new int[]{Types.INTEGER});
- }
- @SuppressWarnings("unchecked")
- public PersonBean queryPerson(int id) {
- /**
- * new PersonRowMapper()是一个实现RowMapper接口的类,
- * 执行回调,实现mapRow()方法将rs对象转换成PersonBean对象返回
- */
- List<PersonBean> pbs = (List<PersonBean>)jdbcTemplate.query("select id,name from person where id = ?", new Object[]{id}, new PersonRowMapper());
- PersonBean pb = null;
- if(pbs.size()>0) {
- pb = pbs.get(0);
- }
- return pb;
- }
- @SuppressWarnings("unchecked")
- public List<PersonBean> queryPersons() {
- List<PersonBean> pbs = (List<PersonBean>) jdbcTemplate.query("select id,name from person", new PersonRowMapper());
- return pbs;
- }
- public void updatePerson(PersonBean person) {
- jdbcTemplate.update("update person set name = ? where id = ?", new Object[]{person.getName(), person.getId()}, new int[]{Types.VARCHAR, Types.INTEGER});
- }
- }
然后我们需要在bean.xml文件中配置:
- <?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:context="http://www.springframework.org/schema/context"
- xmlns:aop="http://www.springframework.org/schema/aop"
- 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.5.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
- <context:property-placeholder location="classpath:jdbc.properties" />
- <bean id="dataSource"
- class="org.apache.commons.dbcp.BasicDataSource"
- destroy-method="close">
- <property name="driverClassName" value="${driverClassName}" />
- <property name="url" value="${url}" />
- <property name="username" value="${username}" />
- <property name="password" value="${password}" />
- <!-- 连接池启动时的初始值 -->
- <property name="initialSize" value="${initialSize}" />
- <!-- 连接池的最大值 -->
- <property name="maxActive" value="${maxActive}" />
- <!-- 最大空闲值.当经过一个高峰时间后,连接池可以慢慢将已经用不到的连接慢慢释放一部分,一直减少到maxIdle为止 -->
- <property name="maxIdle" value="${maxIdle}" />
- <!-- 最小空闲值.当空闲的连接数少于阀值时,连接池就会预申请去一些连接,以免洪峰来时来不及申请 -->
- <property name="minIdle" value="${minIdle}" />
- </bean>
- <bean id="personService"
- class="com.royzhou.jdbc.PersonServiceImpl">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <bean id="txManager"
- class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
- <property name="dataSource" ref="dataSource" />
- </bean>
- <!-- 定义事务传播属性 -->
- <tx:advice id="txAdvice" transaction-manager="txManager">
- <tx:attributes>
- <tx:method name="query*" propagation="NOT_SUPPORTED" read-only="true"/>
- <tx:method name="*" propagation="REQUIRED"/>
- </tx:attributes>
- </tx:advice>
- <aop:config>
- <aop:pointcut id="transactionPointCut" expression="execution(* com.royzhou.jdbc..*.*(..))"/>
- <aop:advisor pointcut-ref="transactionPointCut" advice-ref="txAdvice"/>
- </aop:config>
- </beans>
从新运行测试类
后台抛出RuntimeException异常,查看数据库,没有插入数据,说明事务进行回滚,XML方式的配置也生效了.
另外还有一个编程式的事务处理,但是它有些侵入性。通常我们的事务需求并没有要求在事务的边界上进行如此精确的控制。
我们一般采用"声明式事务"。
总结一下:
事务是企业应用开发的重要组成部分,他使软件更加可靠。它们确保一种要么全有要么全无的行为,防止数据不一致而导致的不可预测的错误发生。
它们同时也支持并发,防止并发应用线程在操作同一数据时互相影响。以前我们写Jdbc代码的时候,可能需要自己手动去开启事务,然后方法执行结束之后
再去提交事务,全部都嵌套在我们的业务代码之中,具有很强的侵入性....
使用Spring提供事务管理机制,我们只需要配置XML或使用Annotion进行注解就可以实现事务的管理和配置,减少了代码之间的耦合,配置也很方便,很大程度上提升了我们的开发效率.
Spring JdbcTemplate 与 事务管理 学习的更多相关文章
- Spring jdbctemplate和事务管理器 全注解配置 不使用xml
/** * spring的配置类,相当于bean.xml */@Configuration//@Configuration标注在类上,相当于把该类作为spring的xml配置文件中的<beans ...
- Spring jdbctemplate和事务管理器
内部bean 对象结构: @Autowiredprivate IAccountService accountService; @Service("accountService")@ ...
- spring事务管理学习
spring事务管理学习 spring的事务管理和mysql自己的事务之间的区别 参考很好介绍事务异常回滚的文章 MyBatis+Spring 事务管理 spring中的事务回滚例子 这篇文章讲解了@ ...
- spring深入学习(五)-----spring dao、事务管理
访问数据库基本是所有java web项目必备的,不论是oracle.mysql,或者是nosql,肯定需要和数据库打交道.一开始学java的时候,肯定是以jdbc为基础,如下: private sta ...
- Spring入门6事务管理2 基于Annotation方式的声明式事务管理机制
Spring入门6事务管理2 基于Annotation方式的声明式事务管理机制 201311.27 代码下载 链接: http://pan.baidu.com/s/1kYc6c 密码: 233t 前言 ...
- Spring入门5.事务管理机制
Spring入门5.事务管理机制 20131126 代码下载 : 链接: http://pan.baidu.com/s/1kYc6c 密码: 233t 回顾之前的知识,Spring 最为核心的两个部分 ...
- (转)使用Spring配置文件实现事务管理
http://blog.csdn.net/yerenyuan_pku/article/details/52886207 前面我们讲解了使用Spring注解方式来管理事务,现在我们就来学习使用Sprin ...
- Spring中的事务管理
事务简介: 事务管理是企业级应用程序开发中必不可少的技术,用来确保数据的完整性和一致性 事务就是一系列的动作,它们被当作一个单独的工作单元.这些动作要么全部完成,要么全部不起作用 事务的四个关键属性( ...
- Spring中的事务管理详解
在这里主要介绍Spring对事务管理的一些理论知识,实战方面参考上一篇博文: http://www.cnblogs.com/longshiyVip/p/5061547.html 1. 事务简介: 事务 ...
随机推荐
- 允许Java App(applet)粘贴方法
修改安全策略文件: "java.policy" JRE6的路径在:"C:\Program Files (x86)\Java\jre6\lib\security" ...
- poj1338 Ugly Numbers 打表, 递推
题意:一个数的质因子能是2, 3, 5, 那么这个数是丑数. 思路: 打表或者递推. 打表: 若该数为丑数,那么一定能被2 或者3, 或者5 整除, 除完之后则为1. #include <ios ...
- Python 基础-3
使用while打印1 2 3 4 5 6 8 9 10 count = 0 #while count < 10: while count < 10: count += 1 if co ...
- k8s 如何 Failover?
上一节我们有 3 个 nginx 副本分别运行在 k8s-node1 和 k8s-node2 上.现在模拟 k8s-node2 故障,关闭该节点. 等待一段时间,Kubernetes 会检查到 k8s ...
- python基础面试题整理---从零开始 每天十题(02)
书接上回,我们继续来说说python的面试题,我在各个网站搜集了一些,我给予你们一个推荐的答案,你们可以组织成自己的语言来说出来,让我们更好的做到面向工资编程 一.Q:说说你对zen of pytho ...
- iOS面试集锦3
1.写一个NSString类的实现 + (id)initWithCString:(c*****t char *)nullTerminatedCString encoding:(NSStringEnco ...
- 博弈论入门 Bash 、Nim 、Wythoff's Game结论及c++代码实现
SG函数先不说,给自己总结下三大博弈.和二进制及黄金分割联系密切,数学真奇妙,如果不用考试就更好了. 1.Bash Game:n个物品,最少取1个,最多取m个,先取完者胜. 给对手留下(m+1)的倍数 ...
- Python使用三种方法实现PCA算法[转]
主成分分析(PCA) vs 多元判别式分析(MDA) PCA和MDA都是线性变换的方法,二者关系密切.在PCA中,我们寻找数据集中最大化方差的成分,在MDA中,我们对类间最大散布的方向更感兴趣. 一句 ...
- Python基础:列表(list)和元组(tuple)
学一门语言,可以用对比其他语言方法加深对这门语言特点的理解. 一.定义:列表和元组,都是一个可以放置任意数据类型的有序集合. mutable的列表:动态的,可以改变元素 immutable的元组:静态 ...
- Python之写入文件(1)
一.写入文件 保存数据也是在各个编程语言中常用的操作,将数据写入到文件中是常用的操作,你可以将程序运行中的一些临时输出保存至文件中,以便后续打开文件查看,也可以把这些文件读入程序中来操作其中的数据. ...