Spring 事务
事务是一组操作的执行单元,对于数据库,事务是一组SQL的执行,事务要求要么全部执行成功,要么全部回滚。
事务特点
Atomic(原子性):要么都发生,要么都不发生。
Consistent(一致性):数据应该不被破坏。
Isolate(隔离性):用户间操作不相混淆。
Durable(持久性):永久保存,例如保存到数据库中等。

事务隔离级别
1、default 默认的事务隔离级别
2、read_uncommitted 未提交读,不能避免脏读,不可重复读,幻读
3、read_committed提交读,不能避免不可重复读,幻读。
4、repeatable_read可重复读,不能避免幻读
5、serializable 序列读,隔离级别最高,能够防止脏读, 不可重复读,幻读。



基于XML配置事务管理

xml
<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:p="http://www.springframework.org/schema/p"
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.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">
<context:component-scan base-package="com.hk.eshop"/>
<!--配置DBCP数据库连接池-->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/test01?useUnicode=true&characterEncoding=utf8"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
<!--连接池中初始化连接数量-->
<property name="initialSize" value="1"/>
<!--连接池中连接数量最大值-->
<property name="maxActive" value="500"/>
<!--连接池中连接最大空闲连接数-->
<property name="maxIdle" value="2"/>
<!--连接池中连接最小空闲连接数-->
<property name="minIdle" value="1"/>
</bean>
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"/>
</bean>
<bean id="userMng" class="com.hk.eshop.jdbc.UserMng">
<property name="jdbcTemplate" ref="jdbcTemplate"/>
</bean>
<aop:aspectj-autoproxy proxy-target-class="true"/>
<bean id="trManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 注解控制事务: 扫描标注事务注解的方法,由事务管理器托管 -->
<!--
<tx:annotation-driven transaction-manager="trManager"/>
-->
<!-- xml控制事务: 扫描标注事务注解的方法,由事务管理器托管 -->
<tx:advice id="txadvice" transaction-manager="trManager">
<tx:attributes>
<tx:method name="save*" isolation="DEFAULT" propagation="REQUIRED" read-only="false" rollback-for="com.hk.eshop.jdbc.UserSaveException"/>
<tx:method name="update*" isolation="DEFAULT" propagation="REQUIRED" read-only="false"/>
<tx:method name="delete*" isolation="DEFAULT" propagation="REQUIRED" read-only="false"/>
<tx:method name="*" read-only="true"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="txpointcut" expression="execution(* com.hk.eshop.jdbc..*.*(..))" />
<aop:advisor advice-ref="txadvice" pointcut-ref="txpointcut"/>
</aop:config>
</beans>





代码片段
java
package com.hk.eshop.jdbc;
import java.sql.Connection;
import java.sql.Statement;
import javax.sql.DataSource;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class TransCtrl {
public static void main(String[] args) throws Exception
{
Connection con = null;
try
{
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
DataSource ds = (DataSource)ac.getBean("dataSource");
con = ds.getConnection();
con.setAutoCommit(false);
Statement stat = con.createStatement();
stat.executeUpdate("update spring_user set name='Java' where id='0'");
con.commit();
}
catch(Exception e)
{
con.rollback();
e.printStackTrace();
}
finally
{
con.close();
}
}
}
package com.hk.eshop.jdbc;
public class UserInfo {
private String id = null;
private String name = null;
private String pwd = null;
private String email = null;
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getPwd() {
return pwd;
}
public void setPwd(String pwd) {
this.pwd = pwd;
}
public String getEmail() {
return email;
}
public void setEmail(String email) {
this.email = email;
}
}
package com.hk.eshop.jdbc;
import java.util.List;
public interface IUserMng {
public void saveUser(UserInfo ui);
public void saveUser2(UserInfo ui);
public void saveUser3(UserInfo ui) throws UserSaveException;
public void updateUser(UserInfo ui);
public void deleteById(String id);
public UserInfo getUser(String id);
public List<UserInfo> find();
}
package com.hk.eshop.jdbc;
public class UserSaveException extends Exception{
public UserSaveException(String msg)
{
super(msg);
}
}
package com.hk.eshop.jdbc;
import org.springframework.dao.EmptyResultDataAccessException;
import org.springframework.jdbc.core.JdbcTemplate;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.List;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.RowMapper;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@Transactional(readOnly=true)
public class UserMng implements IUserMng{
private JdbcTemplate jdbcTemplate;
public JdbcTemplate getJdbcTemplate() {
return jdbcTemplate;
}
public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
this.jdbcTemplate = jdbcTemplate;
}
@Transactional(isolation= Isolation.DEFAULT,propagation= Propagation.REQUIRED,readOnly=false)
public void saveUser(UserInfo ui)
{
String sql = "insert into spring_user(id,name,pwd,email) values (?,?,?,?)";
Object [] args = {ui.getId(),ui.getName(),ui.getPwd(),ui.getEmail()};
int [] argTypes = {java.sql.Types.VARCHAR,java.sql.Types.VARCHAR,
java.sql.Types.VARCHAR,java.sql.Types.VARCHAR};
jdbcTemplate.update(sql, args,argTypes);
}
public void saveUser2(UserInfo ui)
{
try
{
String sql = "insert into spring_user(id,name,pwd,email) values (?,?,?,?)";
Object [] args = {ui.getId(),ui.getName(),ui.getPwd(),ui.getEmail()};
int [] argTypes = {java.sql.Types.VARCHAR,java.sql.Types.VARCHAR,
java.sql.Types.VARCHAR,java.sql.Types.VARCHAR};
jdbcTemplate.update(sql, args,argTypes);
int i = 1/0;//除0操作抛出异常
}
catch(Exception e)
{
throw new RuntimeException("抛出运行时异常!");
}
}
public void saveUser3(UserInfo ui) throws UserSaveException
{
try
{
String sql = "insert into spring_user(id,name,pwd,email) values (?,?,?,?)";
Object [] args = {ui.getId(),ui.getName(),ui.getPwd(),ui.getEmail()};
int [] argTypes = {java.sql.Types.VARCHAR,java.sql.Types.VARCHAR,
java.sql.Types.VARCHAR,java.sql.Types.VARCHAR};
jdbcTemplate.update(sql, args,argTypes);
int i = 1/0;//除0操作抛出异常
}
catch(Exception e)
{
e.printStackTrace();
throw new UserSaveException("user save exception");
}
}
@Transactional(isolation=Isolation.DEFAULT,propagation=Propagation.REQUIRED,readOnly=false)
public void updateUser(UserInfo ui)
{
String sql = "update spring_user set name=?,pwd=?,email=? where id=?";
Object [] args = {ui.getName(),ui.getPwd(),ui.getEmail(),ui.getId()};
jdbcTemplate.update(sql, args);
}
@Transactional(isolation=Isolation.DEFAULT,propagation=Propagation.REQUIRED,readOnly=false)
public void deleteById(String id)
{
String sql = "delete from spring_user where id=?";
Object [] args = {id};
jdbcTemplate.update(sql, args);
}
public UserInfo getUser(String id)
{
String sql = "select id,name,pwd,email from spring_user where id=?";
Object [] args = {id};
RowMapper<UserInfo> rowMapper = new RowMapper<UserInfo>() {
public UserInfo mapRow(ResultSet rs, int rowNum) throws SQLException {
UserInfo ui = new UserInfo();
ui.setId(rs.getString("id"));
ui.setName(rs.getString("name"));
ui.setPwd(rs.getString("pwd"));
ui.setEmail(rs.getString("email"));
return ui;
}
};
try {
UserInfo ui = jdbcTemplate.queryForObject(sql, args, rowMapper);
return ui;
}
catch(EmptyResultDataAccessException e)
{
return null;
}
}
public List<UserInfo> find()
{
String sql = "select id,name,pwd,email from spring_user";
RowMapper<UserInfo> rowMapper = new RowMapper<UserInfo>() {
public UserInfo mapRow(ResultSet rs, int rowNum) throws SQLException {
UserInfo ui = new UserInfo();
ui.setId(rs.getString("id"));
ui.setName(rs.getString("name"));
ui.setPwd(rs.getString("pwd"));
ui.setEmail(rs.getString("email"));
return ui;
}
};
List<UserInfo> list = jdbcTemplate.query(sql, rowMapper);
return list;
}
}
package com.hk.eshop.jdbc;
import java.util.List;
import javax.annotation.Resource;
import javax.sql.DataSource;
import junit.framework.TestCase;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(value=SpringJUnit4ClassRunner.class)
@ContextConfiguration(value="classpath:beans.xml")
public class UserMngTest extends TestCase {
@Resource(name="userMng")
IUserMng userMng;
@Test
public void testDataSource(){
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
DataSource dataSource = (DataSource)ac.getBean("dataSource");
System.out.println("testDataSource dataSource:"+dataSource);
}
@Test
public void saveUser(){
//查询全部表记录
List<UserInfo> lst = userMng.find();
int size = lst.size();
//添加一条数据库记录
UserInfo ui = new UserInfo();
ui.setId("0");
ui.setName("java");
ui.setPwd("pwd");
ui.setEmail("java@test.com");
userMng.saveUser(ui);
List<UserInfo> lst2 = userMng.find();
if(lst2.size() != (size+1)) fail("userMng.saveUser()方法测试失败");
}
@Test
public void updateUser(){
UserInfo ui = new UserInfo();
ui.setId("0");
ui.setName("android");
ui.setPwd("888");
ui.setEmail("android@test.com");
userMng.updateUser(ui);
//查询id=0记录,比较字段是否修改正确
UserInfo chkUi = userMng.getUser("0");
if(!chkUi.getName().equals(ui.getName()) ||
!chkUi.getPwd().equals(ui.getPwd()) ||
!chkUi.getEmail().equals(ui.getEmail()))
{
fail("userMng.updateUser()方法测试失败");
}
}
@Test
public void deleteUser(){
//通过ID删除
String id = "0";
userMng.deleteById(id);
UserInfo chkUi = userMng.getUser(id);
if(chkUi != null) fail("userMng.updateUser()方法测试失败");
}
@Test
public void getUser(){
String id = "0";
UserInfo user = userMng.getUser(id);
System.out.println("getUser user="+user);
}
@Test
public void find(){
List<UserInfo> lst2 = userMng.find();
System.out.println("find lst2.size()="+lst2.size());
}
}
package com.hk.eshop.jdbc;
import junit.framework.TestCase;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import javax.annotation.Resource;
import javax.sql.DataSource;
import java.util.List;
@RunWith(value=SpringJUnit4ClassRunner.class)
@ContextConfiguration(value="classpath:beans.xml")
public class UserMngTest2 extends TestCase {
@Resource(name="userMng")
IUserMng userMng;
@Test
public void saveUser2(){
UserInfo ui = new UserInfo();
ui.setId("0");
ui.setName("java");
ui.setPwd("pwd");
ui.setEmail("java@test.com");
userMng.saveUser2(ui);
System.out.println("saveUser2 ui.name="+ui.getName());
}
}
package com.hk.eshop.jdbc;
import junit.framework.TestCase;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import javax.annotation.Resource;
@RunWith(value=SpringJUnit4ClassRunner.class)
@ContextConfiguration(value="classpath:beans.xml")
public class UserMngTest3 extends TestCase {
@Resource(name="userMng")
IUserMng userMng;
@Test
public void saveUser3() throws UserSaveException{
UserInfo ui = new UserInfo();
ui.setId("0");
ui.setName("java");
ui.setPwd("pwd");
ui.setEmail("java@test.com");
userMng.saveUser3(ui);
System.out.println("saveUser3 ui.name="+ui.getName());
}
}